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.gui.aligpanel;
022
023import java.awt.Color;
024import java.awt.Font;
025import java.awt.Graphics;
026import java.awt.Graphics2D;
027import java.awt.Point;
028import java.awt.Rectangle;
029import java.awt.RenderingHints;
030import java.awt.event.ActionEvent;
031import java.awt.event.WindowEvent;
032import java.awt.event.WindowListener;
033import java.util.ArrayList;
034import java.util.BitSet;
035import java.util.List;
036
037import org.biojava.nbio.structure.Atom;
038import org.biojava.nbio.structure.StructureException;
039import org.biojava.nbio.structure.align.gui.JPrintPanel;
040import org.biojava.nbio.structure.align.gui.MenuCreator;
041import org.biojava.nbio.structure.align.gui.MultipleAlignmentJmolDisplay;
042import org.biojava.nbio.structure.align.gui.jmol.AbstractAlignmentJmol;
043import org.biojava.nbio.structure.align.gui.jmol.JmolTools;
044import org.biojava.nbio.structure.align.model.AFPChain;
045import org.biojava.nbio.structure.align.multiple.MultipleAlignment;
046import org.biojava.nbio.structure.align.multiple.MultipleAlignmentEnsembleImpl;
047import org.biojava.nbio.structure.align.multiple.util.MultipleAlignmentTools;
048import org.biojava.nbio.structure.align.multiple.util.MultipleAlignmentWriter;
049import org.biojava.nbio.structure.align.util.AFPAlignmentDisplay;
050import org.biojava.nbio.structure.gui.events.AlignmentPositionListener;
051import org.biojava.nbio.structure.gui.util.AlignedPosition;
052
053/**
054 * A JPanel that can display the sequence alignment of a
055 * {@link MultipleAlignment} in a nice way and interact with Jmol by
056 * selecting the aligned atoms of the  sequence selection.
057 * <p>
058 * Coloring options include: sequence similarity, by Block or by Structure.
059 * Colors are connected with the JmolPanel using the same pattelete.
060 * <p>
061 * The positions can be selected individually or in ranges and they will be
062 * translated to jmol commands.
063 *
064 * @author Aleix Lafita
065 * @since 4.1.0
066 *
067 */
068public class MultipleAligPanel extends JPrintPanel
069implements AlignmentPositionListener, WindowListener {
070
071        private static final long serialVersionUID = -6892229111166263764L;
072
073        private MultipleAlignment multAln;
074        private List<String> alnSeq; //sequence alignment
075        private List<Integer> mapSeqToStruct;  //mapping from sequence to structure
076
077        int size;                       //number of structures
078        int length;             //number of aligned positions in sequence alignment
079
080        private Font seqFont;
081        private Font eqFont;
082
083        private AbstractAlignmentJmol jmol;
084        private MultipleAligPanelMouseMotionListener mouseMoLi;
085        private MultipleAlignmentCoordManager coordManager;
086
087        private BitSet selection;
088        private boolean selectionLocked;
089
090        private boolean colorBySimilarity=false;
091        private boolean colorByAlignmentBlock=false;
092
093        private static final Color COLOR_EQUAL   = Color.decode("#6A93D4");
094        private static final Color COLOR_SIMILAR = Color.decode("#D460CF");
095
096        /**
097         * Default constructor. Empty MultipleAligPanel instance.
098         */
099        public MultipleAligPanel(){
100                super();
101                this.setBackground(Color.white);
102                seqFont = new Font("SansSerif",Font.PLAIN,12);
103                eqFont = new Font("SansSerif",Font.BOLD,12);
104
105                mouseMoLi = new MultipleAligPanelMouseMotionListener(this);
106                this.addMouseMotionListener(mouseMoLi);
107                this.addMouseListener(mouseMoLi);
108                mouseMoLi.addAligPosListener(this);
109                selection = new BitSet();
110
111                multAln = null;
112                alnSeq = null;
113                mapSeqToStruct = null;
114        }
115
116        /**
117         * Constructor using an afpChain and the atom arrays for pairwise
118         * alignments. The AFPChain is converted into a MultipleAlignment.
119         *
120         * @param afpChain
121         * @param ca1
122         * @param ca2
123         * @throws StructureException
124         */
125        public MultipleAligPanel(AFPChain afpChain, Atom[] ca1, Atom[] ca2,
126                        AbstractAlignmentJmol jmol) throws StructureException {
127
128                this();
129
130                String algorithm = afpChain.getAlgorithmName();
131                boolean flex = false;
132                if (algorithm != null){
133                        if (algorithm.contains("flexible")) flex = true;
134                }
135
136                //Convert the apfChain into a MultipleAlignment object
137                MultipleAlignmentEnsembleImpl ensemble =
138                                new MultipleAlignmentEnsembleImpl(afpChain, ca1, ca2, flex);
139                this.multAln = ensemble.getMultipleAlignment(0);
140
141                //Create the sequence alignment and the structure-sequence mapping.
142                this.mapSeqToStruct = new ArrayList<>();
143                this.alnSeq = MultipleAlignmentTools.getSequenceAlignment(
144                                this.multAln, this.mapSeqToStruct);
145
146                //Initialize other memeber variables of the panel
147                this.size = multAln.size();
148                this.length = alnSeq.get(0).length();
149
150                coordManager = new MultipleAlignmentCoordManager(size, length);
151                this.jmol = jmol;
152        }
153
154        /**
155         * Constructor using a MultipleAlignment.
156         *
157         * @param msa
158         * @param jm
159         */
160        public MultipleAligPanel(MultipleAlignment msa, AbstractAlignmentJmol jm) {
161                this();
162                this.multAln = msa;
163
164                //Create the sequence alignment and the structure-sequence mapping.
165                this.mapSeqToStruct = new ArrayList<>();
166                this.alnSeq = MultipleAlignmentTools.getSequenceAlignment(
167                                this.multAln, this.mapSeqToStruct);
168
169                this.size = multAln.size();
170                this.length = this.alnSeq.get(0).length();
171
172                coordManager = new MultipleAlignmentCoordManager(size, length);
173                this.jmol = jm;
174        }
175
176        public MultipleAlignmentCoordManager getCoordManager() {
177                return coordManager;
178        }
179
180        public void addAlignmentPositionListener(AlignmentPositionListener li){
181                mouseMoLi.addAligPosListener(li);
182        }
183
184        public void destroy(){
185
186                multAln = null;
187                alnSeq = null;
188                mouseMoLi.destroy();
189                jmol = null;
190                selection = null;
191        }
192
193        @Override
194        public void paintComponent(Graphics g){
195
196                super.paintComponent(g);
197
198                Graphics2D g2D = (Graphics2D) g;
199
200                g2D.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING,
201                                RenderingHints.VALUE_TEXT_ANTIALIAS_ON);
202
203                g2D.setRenderingHint(RenderingHints.KEY_ANTIALIASING,
204                                RenderingHints.VALUE_ANTIALIAS_ON);
205
206                int startpos = 0;
207                int endpos = length;
208
209                String summary = multAln.toString();
210                g2D.drawString(summary, 20, coordManager.getSummaryPos());
211
212                Color significantCol = Color.black;
213                //if (multAln.isSignificantResult()) significantCol = Color.green;
214
215                g2D.setPaint(significantCol);
216                Rectangle sig = new Rectangle(10,10,10,10);
217                g2D.fill(sig);
218
219                for (int i = startpos; i < endpos; i++){
220
221                        boolean isGapped = false;
222                        g2D.setFont(seqFont);
223
224                        if (mapSeqToStruct.get(i)!=-1) g2D.setFont(eqFont);
225                        else isGapped = true;
226
227                        //Loop through every structure to get all the points
228                        List<Point> points = new ArrayList<>();
229                        for (int str=0; str<size; str++) points.add(
230                                        coordManager.getPanelPos(str,i));
231                        Point p1 = points.get(0);
232                        Point p2 = points.get(points.size()-1);
233
234                        for (int str=0; str<size; str++){
235
236                                char c = alnSeq.get(str).charAt(i);
237                                Color bg = jmol.getColorPalette().getColorPalette(size)[str];
238
239                                //Color only if the position is aligned
240                                if (!isGapped){
241                                        //Color by sequence similarity (equal or similar)
242                                        if (colorBySimilarity){
243                                                boolean equal = true;
244                                                boolean similar = true;
245                                                char c1 = '-';
246                                                for (int st=0; st<size-1; st++){
247                                                        if (alnSeq.get(st).charAt(i) != '-') {
248                                                                c1 = alnSeq.get(st).charAt(i);
249                                                        }
250                                                        char c2 = alnSeq.get(st+1).charAt(i);
251                                                        //If any position is a gap continue
252                                                        if (c1=='-' || c2=='-' ||
253                                                                        Character.isLowerCase(c1) ||
254                                                                        Character.isLowerCase(c2)) {
255                                                                continue;
256                                                        }
257                                                        if (equal && c1 == c2)
258                                                                continue;
259                                                        else equal = false;
260                                                        if (AFPAlignmentDisplay.aaScore(c1, c2) > 0)
261                                                                continue;
262                                                        else similar = false; break;
263                                                }
264                                                if (equal) bg = COLOR_EQUAL;
265                                                else if (similar) bg = COLOR_SIMILAR;
266                                                else bg = Color.LIGHT_GRAY;
267                                        }
268                                        //Color by alignment block the same way as in the Jmol
269                                        else if (colorByAlignmentBlock){
270                                                int blockNr = MultipleAlignmentTools.
271                                                                getBlockForSequencePosition(
272                                                                                multAln,mapSeqToStruct,i);
273                                                bg = jmol.getColorPalette().getColorPalette(
274                                                                multAln.getBlocks().size())[blockNr];
275                                        }
276                                        if (isSelected(i)) bg = Color.YELLOW;
277
278                                        if (Character.isUpperCase(c) && c!='-'){
279                                                g2D.setPaint(bg);
280                                                Rectangle rec = new Rectangle(points.get(str).x-1,
281                                                                points.get(str).y-11, (p2.x-p1.x)+12,
282                                                                (p2.y-p1.y)/size);
283                                                g2D.fill(rec);
284                                        }
285                                }
286
287                                // draw the AA sequence
288                                g2D.setColor(Color.black);
289                                g2D.drawString(String.valueOf(c), points.get(str).x, points.get(str).y);
290                        }
291                }
292
293                int nrLines = (length-1) /
294                                (MultipleAlignmentCoordManager.DEFAULT_LINE_LENGTH);
295
296                for (int i = 0 ; i < nrLines+1 ; i++){
297
298                        // draw legend at i
299                        for (int str=0; str<size; str++){
300
301                                Point p1 = coordManager.getLegendPosition(i,str);
302
303                                int aligPos = i *
304                                                MultipleAlignmentCoordManager.DEFAULT_LINE_LENGTH;
305                                Atom a1 = null;
306                                while (a1==null &&
307                                                aligPos < Math.min((i+1)*MultipleAlignmentCoordManager.
308                                                                DEFAULT_LINE_LENGTH-1,length)){
309                                        a1 = MultipleAlignmentTools.getAtomForSequencePosition(
310                                                        multAln, mapSeqToStruct, str, aligPos);
311                                        aligPos++;
312                                }
313                                String label1 = JmolTools.getPdbInfo(a1,false);
314                                g2D.drawString(label1, p1.x,p1.y);
315
316                                Point p3 = coordManager.getEndLegendPosition(i,str);
317
318                                aligPos = (i*MultipleAlignmentCoordManager.DEFAULT_LINE_LENGTH+
319                                                MultipleAlignmentCoordManager.DEFAULT_LINE_LENGTH - 1);
320                                if (aligPos > length) aligPos = length-1;
321                                Atom a3 = null;
322                                while (a3==null && aligPos > Math.max(i*
323                                                MultipleAlignmentCoordManager.DEFAULT_LINE_LENGTH,0)){
324                                        a3 = MultipleAlignmentTools.getAtomForSequencePosition(
325                                                        multAln, mapSeqToStruct, str, aligPos);
326                                        aligPos--;
327                                }
328
329                                String label3 = JmolTools.getPdbInfo(a3,false);
330
331                                g2D.drawString(label3, p3.x,p3.y);
332                        }
333                }
334        }
335
336        private boolean isSelected(int alignmentPosition) {
337                return selection.get(alignmentPosition);
338        }
339
340        @Override
341        public void mouseOverPosition(AlignedPosition p) {
342
343                if (!selectionLocked) selection.clear();
344
345                selection.set(p.getPos1());
346                updateJmolDisplay();
347                this.repaint();
348        }
349
350        private void updateJmolDisplay() {
351
352                if (jmol == null) return;
353
354                StringBuffer cmd = new StringBuffer("select ");
355                int nrSelected = 0;
356                for (int i=0; i<length; i++){
357                        if (selection.get(i)){
358                                for (int str=0; str<size; str++){
359                                        Atom a = MultipleAlignmentTools.getAtomForSequencePosition(
360                                                        multAln, mapSeqToStruct,str,i);
361                                        if (a != null) {
362                                                cmd.append(JmolTools.getPdbInfo(a));
363                                                cmd.append("/"+(str+1)+", ");
364                                        }
365                                }
366                                nrSelected++;
367                        }
368                }
369                if (nrSelected == 0) cmd.append(" none;");
370                else cmd.append(" none; set display selected;");
371                //System.out.println(cmd.toString());
372                jmol.evalString(cmd.toString());
373        }
374
375
376        @Override
377        public void positionSelected(AlignedPosition p) {
378                mouseOverPosition(p);
379        }
380
381        @Override
382        public void rangeSelected(AlignedPosition start, AlignedPosition end) {
383
384                if (!selectionLocked) selection.clear();
385                selection.set(start.getPos1(), end.getPos1()+1);
386                updateJmolDisplay();
387                this.repaint();
388        }
389
390        @Override
391        public void selectionLocked() {
392                selectionLocked = true;
393        }
394
395        @Override
396        public void selectionUnlocked() {
397                selectionLocked = false;
398                selection.clear();
399                this.repaint();
400        }
401
402        @Override
403        public void toggleSelection(AlignedPosition p) {
404                selection.flip(p.getPos1());
405                updateJmolDisplay();
406                this.repaint();
407        }
408
409        public void setStructureAlignmentJmol(AbstractAlignmentJmol jmol) {
410                this.jmol = jmol;
411
412        }
413
414        @Override
415        public void windowActivated(WindowEvent e) {}
416
417        @Override
418        public void windowClosed(WindowEvent e) {}
419
420        @Override
421        public void windowClosing(WindowEvent e) {
422                destroy();
423        }
424
425        @Override
426        public void windowDeactivated(WindowEvent e) {}
427
428        @Override
429        public void windowDeiconified(WindowEvent e) {}
430
431        @Override
432        public void windowIconified(WindowEvent e) {}
433
434        @Override
435        public void windowOpened(WindowEvent e) {}
436
437        @Override
438        public void actionPerformed(ActionEvent e) {
439                String cmd = e.getActionCommand();
440                if ( cmd.equals(MenuCreator.PRINT)) {
441                        super.actionPerformed(e);
442                } else if (cmd.equals(MenuCreator.FASTA_FORMAT)){
443                        String result = MultipleAlignmentWriter.toFASTA(multAln);
444                        MultipleAlignmentJmolDisplay.showAlignmentImage(multAln, result);
445                } else if ( cmd.equals(MenuCreator.PAIRS_ONLY)) {
446                        String result = MultipleAlignmentWriter.toAlignedResidues(multAln);
447                        MultipleAlignmentJmolDisplay.showAlignmentImage(multAln, result);
448                } else if (cmd.equals(MenuCreator.FATCAT_TEXT)){
449                        String result = MultipleAlignmentWriter.toFatCat(multAln);
450                        MultipleAlignmentJmolDisplay.showAlignmentImage(multAln, result);
451                } else if (cmd.equals(MenuCreator.SELECT_EQR)){
452                        selectEQR();
453                } else if ( cmd.equals(MenuCreator.SIMILARITY_COLOR)){
454                        colorBySimilarity(true);
455                } else if (cmd.equals(MenuCreator.EQR_COLOR)){
456                        colorBySimilarity(false);
457                } else if ( cmd.equals(MenuCreator.FATCAT_BLOCK)){
458                        colorByAlignmentBlock();
459                } else {
460                        System.err.println("Unknown command:" + cmd);
461                }
462        }
463
464        private void selectEQR() {
465
466                selection.clear();
467
468                for (int pos=0; pos<length; pos++){
469                        if (mapSeqToStruct.get(pos)!=-1) selection.flip(pos);
470                }
471                mouseMoLi.triggerSelectionLocked(true);
472                updateJmolDisplay();
473                this.repaint();
474        }
475
476        private void colorByAlignmentBlock() {
477                colorByAlignmentBlock = true;
478                colorBySimilarity = false;
479                this.repaint();
480        }
481
482        private void colorBySimilarity(boolean flag) {
483                this.colorBySimilarity = flag;
484                colorByAlignmentBlock = false;
485                this.repaint();
486        }
487
488        public List<Atom[]> getAtomArrays() {
489                return multAln.getAtomArrays();
490        }
491        public MultipleAlignment getMultipleAlignment(){
492                return multAln;
493        }
494        public List<String> getAlnSequences(){
495                return alnSeq;
496        }
497        public List<Integer> getMapSeqToStruct(){
498                return mapSeqToStruct;
499        }
500}