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 java.util.ArrayList;
024import java.util.List;
025
026import javax.vecmath.Matrix4d;
027
028import org.biojava.nbio.structure.Atom;
029import org.biojava.nbio.structure.Calc;
030import org.biojava.nbio.structure.StructureException;
031import org.biojava.nbio.structure.StructureTools;
032import org.biojava.nbio.structure.align.multiple.Block;
033import org.biojava.nbio.structure.align.multiple.BlockSet;
034import org.biojava.nbio.structure.align.multiple.MultipleAlignment;
035import org.biojava.nbio.structure.geometry.SuperPositionSVD;
036import org.biojava.nbio.structure.geometry.SuperPositions;
037
038/**
039 * Superimposes each structure in a {@link MultipleAlignment} onto a reference
040 * structure.
041 * <p>
042 * Performs a global superposition of the MultipleAlignment in case
043 * there is only one {@link BlockSet}, and a superposition for every BlockSet
044 * in case there is more than one (flexible alignment).
045 * <p>
046 * This class uses the {@link SuperPositionSVD} algorithm.
047 *
048 * @author Spencer Bliven
049 * @author Aleix Lafita
050 * @since 4.1.0
051 *
052 */
053public class ReferenceSuperimposer implements MultipleSuperimposer {
054
055        private int reference;
056
057        /**
058         * Default Constructor.
059         * Uses the first structure as the reference.
060         */
061        public ReferenceSuperimposer() {
062                this(0);
063        }
064
065        /**
066         * Constructor using a specified structure as reference.
067         *
068         * @param reference Index of the structure to use as a reference
069         *                      (it has to be > 0)
070         */
071        public ReferenceSuperimposer(int reference) {
072                if (reference<0) {
073                        throw new IllegalArgumentException(
074                                        "reference index has to be positive, but was "+reference);
075                }
076                this.reference = reference;
077        }
078
079        @Override
080        public void superimpose(MultipleAlignment alignment)
081                        throws StructureException {
082
083                //Check for inconsistencies in the alignment
084                if(alignment.getEnsemble() == null) {
085                        throw new NullPointerException("No ensemble set for this alignment."
086                                        + " Structure information cannot be obtained.");
087                }
088                if (alignment.size() < 1) {
089                        throw new IndexOutOfBoundsException(
090                                        "No aligned structures, alignment size == 0.");
091                }
092                if (alignment.getCoreLength() < 1){
093                        throw new IndexOutOfBoundsException(
094                                        "Alignment too short, core alignment length < 1.");
095                }
096
097                List<Atom[]> atomArrays = alignment.getAtomArrays();
098                if (atomArrays.size() <= reference) {
099                        throw new IndexOutOfBoundsException(String.format(
100                                        "Invalid reference structure: requested %d but "
101                                                        + "only %d structures.",
102                                                        reference,atomArrays.size()));
103                }
104
105                alignment.clear();
106
107                //Calculate BlockSet transformations
108                for (BlockSet bs:alignment.getBlockSets()){
109
110                        //Block transformations
111                        List<Matrix4d> transforms =
112                                        new ArrayList<Matrix4d>(atomArrays.size());
113
114                        //Loop through structures
115                        for (int i=0; i<atomArrays.size(); i++){
116
117                                if( i == reference) {
118                                        //Identity operation
119                                        Matrix4d ident = new Matrix4d();
120                                        ident.setIdentity();
121                                        transforms.add(ident);
122                                        continue;
123                                }
124
125                                Atom[] ref = atomArrays.get(reference);
126                                Atom[] curr = atomArrays.get(i);
127
128                                List<Atom> atomSet1 = new ArrayList<>();
129                                List<Atom> atomSet2 = new ArrayList<>();
130
131                                for( Block blk : bs.getBlocks() ) {
132                                        if( blk.size() != atomArrays.size()) {
133                                                throw new IllegalStateException(String.format(
134                                                                "Mismatched block length. Expected %d "
135                                                                                + "structures, found %d.",
136                                                                                atomArrays.size(),blk.size() ));
137                                        }
138                                        //Loop through all aligned residues
139                                        for (int j=0; j<blk.length(); j++){
140                                                Integer pos1 = blk.getAlignRes().get(reference).get(j);
141                                                Integer pos2 = blk.getAlignRes().get(i).get(j);
142
143                                                if (pos1==null || pos2==null) continue;
144                                                atomSet1.add(ref[pos1]);
145                                                atomSet2.add(curr[pos2]);
146                                        }
147                                }
148                                Atom[] array1 = atomSet1.toArray(new Atom[atomSet1.size()]);
149                                Atom[] array2 = atomSet2.toArray(new Atom[atomSet2.size()]);
150
151                                array2 = StructureTools.cloneAtomArray(array2);
152
153                                //From the superimposer we obtain the rotation and translation
154                                Matrix4d trans = SuperPositions.superpose(Calc.atomsToPoints(array1),
155                                                Calc.atomsToPoints(array2));
156                                transforms.add(trans);
157                        }
158                        //Set transformation of the BlockSet
159                        bs.setTransformations(transforms);
160                }
161        }
162}