Merge branch 'bug/JAL-2865removeGC' into develop
[jalview.git] / src / jalview / gui / OverviewCanvas.java
1 /*
2  * Jalview - A Sequence Alignment Editor and Viewer ($$Version-Rel$$)
3  * Copyright (C) $$Year-Rel$$ The Jalview Authors
4  * 
5  * This file is part of Jalview.
6  * 
7  * Jalview is free software: you can redistribute it and/or
8  * modify it under the terms of the GNU General Public License 
9  * as published by the Free Software Foundation, either version 3
10  * of the License, or (at your option) any later version.
11  *  
12  * Jalview is distributed in the hope that it will be useful, but 
13  * WITHOUT ANY WARRANTY; without even the implied warranty 
14  * of MERCHANTABILITY or FITNESS FOR A PARTICULAR 
15  * PURPOSE.  See the GNU General Public License for more details.
16  * 
17  * You should have received a copy of the GNU General Public License
18  * along with Jalview.  If not, see <http://www.gnu.org/licenses/>.
19  * The Jalview Authors are detailed in the 'AUTHORS' file.
20  */
21 package jalview.gui;
22
23 import jalview.api.AlignViewportI;
24 import jalview.bin.Cache;
25 import jalview.renderer.OverviewRenderer;
26 import jalview.renderer.OverviewResColourFinder;
27 import jalview.viewmodel.OverviewDimensions;
28
29 import java.awt.Color;
30 import java.awt.Dimension;
31 import java.awt.Graphics;
32 import java.awt.image.BufferedImage;
33
34 import javax.swing.JComponent;
35
36 public class OverviewCanvas extends JComponent
37 {
38   private static final Color TRANS_GREY = new Color(100, 100, 100, 25);
39
40   // This is set true if the alignment view changes whilst
41   // the overview is being calculated
42   private volatile boolean restart = false;
43
44   private volatile boolean updaterunning = false;
45
46   private boolean dispose = false;
47
48   private BufferedImage miniMe;
49
50   private BufferedImage lastMiniMe = null;
51
52   // Can set different properties in this seqCanvas than
53   // main visible SeqCanvas
54   private SequenceRenderer sr;
55
56   private jalview.renderer.seqfeatures.FeatureRenderer fr;
57
58   private OverviewDimensions od;
59
60   private OverviewRenderer or = null;
61
62   private AlignViewportI av;
63
64   private OverviewResColourFinder cf;
65
66   private ProgressPanel progressPanel;
67
68   public OverviewCanvas(OverviewDimensions overviewDims,
69           AlignViewportI alignvp, ProgressPanel pp)
70   {
71     od = overviewDims;
72     av = alignvp;
73     progressPanel = pp;
74
75     sr = new SequenceRenderer(av);
76     sr.renderGaps = false;
77     fr = new jalview.renderer.seqfeatures.FeatureRenderer(av);
78
79     boolean useLegacy = Cache.getDefault(Preferences.USE_LEGACY_GAP, false);
80     Color gapCol = Cache.getDefaultColour(Preferences.GAP_COLOUR,
81             jalview.renderer.OverviewResColourFinder.OVERVIEW_DEFAULT_GAP);
82     Color hiddenCol = Cache.getDefaultColour(Preferences.HIDDEN_COLOUR,
83             jalview.renderer.OverviewResColourFinder.OVERVIEW_DEFAULT_HIDDEN);
84     cf = new OverviewResColourFinder(useLegacy, gapCol, hiddenCol);
85
86     setSize(od.getWidth(), od.getHeight());
87   }
88
89   /**
90    * Update the overview dimensions object used by the canvas (e.g. if we change
91    * from showing hidden columns to hiding them or vice versa)
92    * 
93    * @param overviewDims
94    */
95   public void resetOviewDims(OverviewDimensions overviewDims)
96   {
97     od = overviewDims;
98   }
99
100   /**
101    * Signals to drawing code that the associated alignment viewport has changed
102    * and a redraw will be required
103    */
104   public boolean restartDraw()
105   {
106     synchronized (this)
107     {
108       if (updaterunning)
109       {
110         restart = true;
111         if (or != null)
112         {
113           or.setRedraw(true);
114         }
115       }
116       else
117       {
118         updaterunning = true;
119       }
120       return restart;
121     }
122   }
123
124   /**
125    * Draw the overview sequences
126    * 
127    * @param showSequenceFeatures
128    *          true if sequence features are to be shown
129    * @param showAnnotation
130    *          true if the annotation is to be shown
131    * @param transferRenderer
132    *          the renderer to transfer feature colouring from
133    */
134   public void draw(boolean showSequenceFeatures, boolean showAnnotation,
135           FeatureRenderer transferRenderer)
136   {
137     miniMe = null;
138
139     if (showSequenceFeatures)
140     {
141       fr.transferSettings(transferRenderer);
142     }
143
144     setPreferredSize(new Dimension(od.getWidth(), od.getHeight()));
145
146     or = new OverviewRenderer(fr, od, av.getAlignment(),
147             av.getResidueShading(), cf);
148
149     or.addPropertyChangeListener(progressPanel);
150
151     miniMe = or.draw(od.getRows(av.getAlignment()),
152             od.getColumns(av.getAlignment()));
153
154     Graphics mg = miniMe.getGraphics();
155
156     if (showAnnotation)
157     {
158       mg.translate(0, od.getSequencesHeight());
159       or.drawGraph(mg, av.getAlignmentConservationAnnotation(),
160               av.getCharWidth(), od.getGraphHeight(),
161               od.getColumns(av.getAlignment()));
162       mg.translate(0, -od.getSequencesHeight());
163     }
164
165     or.removePropertyChangeListener(progressPanel);
166     or = null;
167     if (restart)
168     {
169       restart = false;
170       if (!dispose)
171       {
172         draw(showSequenceFeatures, showAnnotation, transferRenderer);
173       }
174     }
175     else
176     {
177       updaterunning = false;
178       lastMiniMe = miniMe;
179     }
180   }
181
182   @Override
183   public void paintComponent(Graphics g)
184   {
185     super.paintComponent(g);
186
187     if (restart)
188     {
189       if (lastMiniMe == null)
190       {
191         g.setColor(Color.white);
192         g.fillRect(0, 0, getWidth(), getHeight());
193       }
194       else
195       {
196         g.drawImage(lastMiniMe, 0, 0, getWidth(), getHeight(), this);
197       }
198       g.setColor(TRANS_GREY);
199       g.fillRect(0, 0, getWidth(), getHeight());
200     }
201     else if (lastMiniMe != null)
202     {
203       // is this a resize?
204       if ((getWidth() > 0) && (getHeight() > 0)
205               && ((getWidth() != od.getWidth())
206                       || (getHeight() != od.getHeight())))
207       {
208         // if there is annotation, scale the alignment and annotation
209         // separately
210         if (od.getGraphHeight() > 0)
211         {
212           BufferedImage topImage = lastMiniMe.getSubimage(0, 0,
213                   od.getWidth(), od.getSequencesHeight());
214           BufferedImage bottomImage = lastMiniMe.getSubimage(0,
215                   od.getSequencesHeight(), od.getWidth(),
216                   od.getGraphHeight());
217
218           // must be done at this point as we rely on using old width/height
219           // above, and new width/height below
220           od.setWidth(getWidth());
221           od.setHeight(getHeight());
222
223           // stick the images back together so lastMiniMe is consistent in the
224           // event of a repaint - BUT probably not thread safe
225           lastMiniMe = new BufferedImage(od.getWidth(), od.getHeight(),
226                   BufferedImage.TYPE_INT_RGB);
227           Graphics lg = lastMiniMe.getGraphics();
228           lg.drawImage(topImage, 0, 0, od.getWidth(),
229                   od.getSequencesHeight(), null);
230           lg.drawImage(bottomImage, 0, od.getSequencesHeight(),
231                   od.getWidth(), od.getGraphHeight(), this);
232           lg.dispose();
233         }
234         else
235         {
236           od.setWidth(getWidth());
237           od.setHeight(getHeight());
238         }
239
240         // make sure the box is in the right place
241         od.setBoxPosition(av.getAlignment().getHiddenSequences(),
242                 av.getAlignment().getHiddenColumns());
243       }
244       // fall back to normal behaviour
245       g.drawImage(lastMiniMe, 0, 0, getWidth(), getHeight(), this);
246     }
247     else
248     {
249       g.drawImage(lastMiniMe, 0, 0, getWidth(), getHeight(), this);
250     }
251     
252     // draw the box
253     g.setColor(Color.red);
254     od.drawBox(g);
255   }
256
257
258   public void dispose()
259   {
260     dispose = true;
261     synchronized (this)
262     {
263       restart = true;
264       if (or != null)
265       {
266         or.setRedraw(true);
267       }
268     }
269   }
270 }