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.align.multiple.util; 022 023import org.biojava.nbio.structure.*; 024import org.biojava.nbio.structure.align.multiple.MultipleAlignment; 025import org.slf4j.Logger; 026import org.slf4j.LoggerFactory; 027 028import javax.vecmath.Matrix4d; 029 030import java.util.ArrayList; 031import java.util.Arrays; 032import java.util.List; 033 034/** 035 * Utility functions to generalize the visualization of MultipleAlignments in 036 * molecular viewers. The methods return different types of selectors for the 037 * aligned residues in the alignment. 038 * 039 * @author Andreas Prlic 040 * @author Aleix Lafita 041 * @author Spencer Bliven 042 * @since 4.2.0 043 * 044 */ 045public class MultipleAlignmentDisplay { 046 047 private static final Logger logger = LoggerFactory 048 .getLogger(MultipleAlignmentDisplay.class); 049 050 /** 051 * New structures are downloaded if they were not cached in the alignment 052 * and they are entirely transformed here with the superposition information 053 * in the Multiple Alignment. 054 * 055 * @param multAln 056 * @return list of transformed AtomArrays 057 * @throws StructureException 058 */ 059 public static List<Atom[]> getRotatedAtoms(MultipleAlignment multAln) 060 throws StructureException { 061 062 int size = multAln.size(); 063 064 List<Atom[]> atomArrays = multAln.getAtomArrays(); 065 for (int i = 0; i < size; i++) { 066 if (atomArrays.get(i).length < 1) 067 throw new StructureException( 068 "Length of atoms arrays is too short! Size: " 069 + atomArrays.get(i).length); 070 } 071 072 List<Atom[]> rotatedAtoms = new ArrayList<>(); 073 074 // TODO implement independent BlockSet superposition of the structure 075 List<Matrix4d> transf = multAln.getBlockSet(0).getTransformations(); 076 077 if (transf == null) { 078 079 logger.error("Alignment Transformations are not calculated. " 080 + "Superimposing to first structure as reference."); 081 082 multAln = multAln.clone(); 083 MultipleSuperimposer imposer = new ReferenceSuperimposer(); 084 imposer.superimpose(multAln); 085 transf = multAln.getBlockSet(0).getTransformations(); 086 assert (transf != null); 087 } 088 089 // Rotate the atom coordinates of all the structures 090 for (int i = 0; i < size; i++) { 091 // TODO handle BlockSet-level transformations 092 // make sure this method has the same behavior as the other display. 093 // -SB 2015-06 094 095 // Assume all atoms are from the same structure 096 Structure displayS = atomArrays.get(i)[0].getGroup().getChain() 097 .getStructure().clone(); 098 099 // Get all the atoms and include ligands and hetatoms 100 Atom[] rotCA = StructureTools.getRepresentativeAtomArray(displayS); 101 List<Group> hetatms = StructureTools.getUnalignedGroups(rotCA); 102 int index = rotCA.length; 103 rotCA = Arrays.copyOf(rotCA, rotCA.length + hetatms.size()); 104 for (Group g : hetatms) { 105 rotCA[index] = g.getAtom(0); 106 index++; 107 } 108 109 // Transform the structure to ensure a full rotation in the display 110 Calc.transform(displayS, transf.get(i)); 111 rotatedAtoms.add(rotCA); 112 } 113 114 return rotatedAtoms; 115 } 116}