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.gui; 022 023import java.awt.event.KeyEvent; 024import java.util.ArrayList; 025import java.util.Arrays; 026import java.util.List; 027 028import javax.swing.JMenu; 029import javax.swing.JMenuBar; 030import javax.swing.JMenuItem; 031 032import org.biojava.nbio.structure.Atom; 033import org.biojava.nbio.structure.Group; 034import org.biojava.nbio.structure.StructureException; 035import org.biojava.nbio.structure.StructureTools; 036import org.biojava.nbio.structure.align.gui.MultipleAlignmentJmolDisplay; 037import org.biojava.nbio.structure.align.gui.StructureAlignmentDisplay; 038import org.biojava.nbio.structure.align.gui.jmol.AbstractAlignmentJmol; 039import org.biojava.nbio.structure.align.gui.jmol.MultipleAlignmentJmol; 040import org.biojava.nbio.structure.align.multiple.MultipleAlignment; 041import org.biojava.nbio.structure.align.util.RotationAxis; 042import org.biojava.nbio.structure.symmetry.axis.AxisAligner; 043import org.biojava.nbio.structure.symmetry.core.QuatSymmetryResults; 044import org.biojava.nbio.structure.symmetry.internal.CeSymmResult; 045import org.biojava.nbio.structure.symmetry.internal.SymmetryAxes; 046import org.biojava.nbio.structure.symmetry.internal.SymmetryAxes.Axis; 047import org.biojava.nbio.structure.symmetry.jmolScript.JmolSymmetryScriptGenerator; 048import org.biojava.nbio.structure.symmetry.jmolScript.JmolSymmetryScriptGeneratorPointGroup; 049import org.biojava.nbio.structure.symmetry.utils.SymmetryTools; 050import org.slf4j.Logger; 051import org.slf4j.LoggerFactory; 052 053/** 054 * Class that provides visualizations methods for symmetry alignments. Call the 055 * display() method for the default visualization of symmetry. 056 * 057 * @author Aleix Lafita 058 * @since 4.2.0 059 * 060 */ 061public class SymmetryDisplay { 062 063 private static final Logger logger = LoggerFactory 064 .getLogger(SymmetryDisplay.class); 065 066 /** 067 * Displays a multiple alignment of the symmetry repeats. 068 * 069 * * @param symm CeSymmResult 070 * 071 * @throws StructureException 072 */ 073 public static MultipleAlignmentJmol displayRepeats(CeSymmResult symm) 074 throws StructureException { 075 076 MultipleAlignment repeats = SymmetryTools.toRepeatsAlignment(symm); 077 MultipleAlignmentJmol jmol = MultipleAlignmentJmolDisplay 078 .display(repeats); 079 jmol.setTitle(getSymmTitle(symm)); 080 return jmol; 081 } 082 083 /** 084 * Displays a multiple alignment of the whole structure transformations 085 * colored by blocks, corresponding to the symmetric protodomains. 086 * 087 * @param symm 088 * CeSymmResult 089 * @throws StructureException 090 */ 091 public static MultipleAlignmentJmol displayFull(CeSymmResult symm) 092 throws StructureException { 093 094 MultipleAlignment full = SymmetryTools.toFullAlignment(symm); 095 096 MultipleAlignmentJmol jmol = MultipleAlignmentJmolDisplay.display(full); 097 jmol.setColorByBlocks(true); 098 jmol.setTitle(getSymmTitle(symm)); 099 100 return jmol; 101 } 102 103 /** 104 * Displays a single structure in a cartoon representation with each 105 * symmetric repeat colored differently. 106 * 107 * @param symmResult 108 * the symmetry multiple alignment obtained from CeSymm 109 * @throws StructureException 110 */ 111 public static AbstractAlignmentJmol display(CeSymmResult symmResult) 112 throws StructureException { 113 114 if (symmResult.isSignificant() && symmResult.isRefined()) { 115 // Show the structure colored by repeat (do not rotate) 116 MultipleAlignment msa = symmResult.getMultipleAlignment(); 117 List<Atom[]> atoms = msa.getAtomArrays(); 118 119 // Add non polymer protein groups 120 Atom[] allAtoms = atoms.get(0); 121 List<Group> hetatms = StructureTools.getUnalignedGroups(allAtoms); 122 allAtoms = Arrays 123 .copyOf(allAtoms, allAtoms.length + hetatms.size()); 124 for (int h = 0; h < hetatms.size(); h++) { 125 int index = (allAtoms.length - hetatms.size()) + h; 126 allAtoms[index] = hetatms.get(h).getAtom(0); 127 } 128 for (int s = 0; s < msa.size(); s++) 129 atoms.set(s, allAtoms); 130 131 MultipleAlignmentJmol jmol = new MultipleAlignmentJmol(msa, atoms); 132 jmol.setTitle(jmol.getStructure().getPDBHeader().getTitle()); 133 addSymmetryMenu(jmol, symmResult); 134 jmol.evalString(printSymmetryGroup(symmResult)); 135 jmol.evalString(printSymmetryAxes(symmResult)); 136 jmol.setTitle(getSymmTitle(symmResult)); 137 jmol.evalString("save STATE state_1"); 138 return jmol; 139 } else { 140 // Show the optimal self-alignment 141 logger.info("Showing optimal self-alignment"); 142 Atom[] cloned = StructureTools 143 .cloneAtomArray(symmResult.getAtoms()); 144 AbstractAlignmentJmol jmol = StructureAlignmentDisplay.display( 145 symmResult.getSelfAlignment(), symmResult.getAtoms(), 146 cloned); 147 RotationAxis axis = new RotationAxis(symmResult.getSelfAlignment()); 148 jmol.evalString(axis.getJmolScript(symmResult.getAtoms())); 149 jmol.evalString("save STATE state_1"); 150 return jmol; 151 } 152 } 153 154 /** 155 * Adds a Symmetry menu to the Jmol display, so that further symmetry 156 * analysis can be triggered. 157 * 158 * @param jmol 159 * parent jmol 160 * @param symmResult 161 * CeSymmResult 162 */ 163 private static void addSymmetryMenu(MultipleAlignmentJmol jmol, 164 CeSymmResult symmResult) { 165 166 JMenuBar menubar = jmol.getFrame().getJMenuBar(); 167 168 JMenu symm = new JMenu("Symmetry"); 169 symm.setMnemonic(KeyEvent.VK_S); 170 171 SymmetryListener li = new SymmetryListener(jmol, symmResult); 172 173 JMenuItem repeats = new JMenuItem("Repeats Superposition"); 174 repeats.addActionListener(li); 175 symm.add(repeats); 176 177 JMenuItem multiple = new JMenuItem("Multiple Structure Alignment"); 178 multiple.addActionListener(li); 179 symm.add(multiple); 180 181 JMenuItem self = new JMenuItem("Optimal Self Alignment"); 182 self.addActionListener(li); 183 symm.add(self); 184 185 JMenuItem pg = new JMenuItem("Show Symmetry Group"); 186 pg.addActionListener(li); 187 symm.add(pg); 188 189 JMenuItem ax = new JMenuItem("Show Symmetry Axes"); 190 ax.addActionListener(li); 191 symm.add(ax); 192 193 JMenuItem news = new JMenuItem("New Symmetry Analysis"); 194 news.addActionListener(li); 195 symm.add(news); 196 197 menubar.add(symm, 3); 198 jmol.getFrame().pack(); 199 } 200 201 /** 202 * Generates a String that displays the symmetry axes of a structure. 203 * 204 * @param symm 205 * CeSymmResult 206 * @return 207 * @throws StructureException 208 */ 209 public static String printSymmetryAxes(CeSymmResult symm) 210 throws StructureException { 211 return printSymmetryAxes(symm,true); 212 } 213 214 /** 215 * Generates a String that displays the symmetry axes of a structure. 216 * 217 * @param symm 218 * CeSymmResult 219 * @param allAxes Indicates whether all axes should be displayed or just 220 * the elemenatary ones 221 * @return 222 * @throws StructureException 223 */ 224 public static String printSymmetryAxes(CeSymmResult symm,boolean allAxes) 225 throws StructureException { 226 227 int id = 0; 228 String script = ""; 229 SymmetryAxes axes = symm.getAxes(); 230 List<Atom[]> repeats = SymmetryTools.toRepeatsAlignment(symm) 231 .getAtomArrays(); 232 233 List<Axis> symmAxes; 234 if(allAxes) { 235 symmAxes = axes.getSymmetryAxes(); 236 } else { 237 symmAxes= axes.getElementaryAxesObjects(); 238 } 239 for (Axis a : symmAxes) { 240 RotationAxis rot = a.getRotationAxis(); 241 List<List<Integer>> cyclicForm = axes.getRepeatsCyclicForm(a); 242 List<Atom> repAtoms = new ArrayList<>(); 243 for(List<Integer> cycle : cyclicForm) { 244 for(Integer repeat : cycle) { 245 repAtoms.addAll(Arrays.asList(repeats.get(repeat))); 246 } 247 } 248 249 script += rot.getJmolScript( 250 repAtoms.toArray(new Atom[repAtoms.size()]), id); 251 id++; 252 } 253 254 return script; 255 } 256 257 /** 258 * Given a symmetry alignment, it draws the symmetry group axes and the 259 * polyhedron box around the structure. It uses the quaternary symmetry 260 * detection code, but tries to factor out the alignment and detection 261 * steps. 262 * 263 * @param symm 264 * CeSymmResult 265 * @return 266 * @throws StructureException 267 */ 268 public static String printSymmetryGroup(CeSymmResult symm) 269 throws StructureException { 270 271 QuatSymmetryResults gSymmetry = SymmetryTools 272 .getQuaternarySymmetry(symm); 273 274 AxisAligner axes = AxisAligner.getInstance(gSymmetry); 275 276 // Draw the axes as in the quaternary symmetry 277 JmolSymmetryScriptGenerator scriptGenerator = JmolSymmetryScriptGeneratorPointGroup 278 .getInstance(axes, "g"); 279 280 String script = "save selection; set measurementUnits ANGSTROMS;" 281 + "select all; set antialiasDisplay true; autobond=false; "; 282 283 script += scriptGenerator.getInstantaneousOrientation(0); 284 script += "restore selection; "; 285 script += scriptGenerator.drawPolyhedron(); 286 script += scriptGenerator.drawAxes(); 287 script += "draw axes* on; draw poly* on; "; 288 289 return script; 290 } 291 292 /** 293 * Create a symmetry title for a display frame (Jmol, alignment, etc). The 294 * title contains information about the algorithm, structure id and 295 * parameters used. 296 * 297 * @param result 298 * @return title String 299 */ 300 public static String getSymmTitle(CeSymmResult result) { 301 302 StringBuffer buff = new StringBuffer(); 303 304 // Add algorithm name and version 305 buff.append(result.getMultipleAlignment().getEnsemble() 306 .getAlgorithmName()); 307 buff.append(" V"); 308 buff.append(result.getMultipleAlignment().getEnsemble().getVersion()); 309 buff.append(": "); 310 311 // Add the result summary string 312 buff.append(result.toString()); 313 return buff.toString(); 314 } 315 316}