Fastest implementation so far
[jalview.git] / src / jalview / appletgui / OverviewPanel.java
1 /*\r
2  * Jalview - A Sequence Alignment Editor and Viewer\r
3  * Copyright (C) 2005 AM Waterhouse, J Procter, G Barton, M Clamp, S Searle\r
4  *\r
5  * This program is free software; you can redistribute it and/or\r
6  * modify it under the terms of the GNU General Public License\r
7  * as published by the Free Software Foundation; either version 2\r
8  * of the License, or (at your option) any later version.\r
9  *\r
10  * This program is distributed in the hope that it will be useful,\r
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of\r
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\r
13  * GNU General Public License for more details.\r
14  *\r
15  * You should have received a copy of the GNU General Public License\r
16  * along with this program; if not, write to the Free Software\r
17  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA\r
18  */\r
19 \r
20 package jalview.appletgui;\r
21 \r
22 import java.awt.*;\r
23 import java.awt.event.*;\r
24 \r
25 public class OverviewPanel\r
26     extends Panel implements Runnable, MouseMotionListener, MouseListener\r
27 {\r
28   Image miniMe;\r
29   AlignViewport av;\r
30   AlignmentPanel ap;\r
31   float scalew = 1f;\r
32   float scaleh = 1f;\r
33 \r
34   public int width, sequencesHeight;\r
35   int graphHeight = 20;\r
36   int boxX = -1, boxY = -1, boxWidth = -1, boxHeight = -1;\r
37 \r
38   boolean resizing = false;\r
39 \r
40   // Can set different properties in this seqCanvas than\r
41   // main visible SeqCanvas\r
42   SequenceRenderer sr;\r
43   FeatureRenderer fr;\r
44 \r
45 \r
46   Frame nullFrame;\r
47 \r
48   public OverviewPanel(AlignmentPanel ap)\r
49   {\r
50     this.av = ap.av;\r
51     this.ap = ap;\r
52     setLayout(null);\r
53     nullFrame = new Frame();\r
54     nullFrame.addNotify();\r
55 \r
56 \r
57     sr = new SequenceRenderer(av);\r
58     sr.graphics = nullFrame.getGraphics();\r
59     sr.renderGaps( false );\r
60     sr.forOverview = true;\r
61     fr = new FeatureRenderer(av);\r
62     fr.overview = true;\r
63 \r
64 \r
65 \r
66     // scale the initial size of overviewpanel to shape of alignment\r
67     float initialScale = (float) av.alignment.getWidth() /\r
68         (float) av.alignment.getHeight();\r
69 \r
70     if(av.vconsensus==null)\r
71           graphHeight = 0;\r
72 \r
73     if (av.alignment.getWidth() > av.alignment.getHeight())\r
74     {\r
75       // wider\r
76       width = 400;\r
77       sequencesHeight = (int) (400f / initialScale);\r
78       if(sequencesHeight<40)\r
79               sequencesHeight = 40;\r
80     }\r
81     else\r
82     {\r
83       // taller\r
84       width = (int) (400f * initialScale);\r
85       sequencesHeight = 300;\r
86       if (width < 120)\r
87       {\r
88         width = 120;\r
89       }\r
90     }\r
91 \r
92     setSize(new Dimension(width, sequencesHeight + graphHeight));\r
93     addComponentListener(new ComponentAdapter()\r
94     {\r
95 \r
96       public void componentResized(ComponentEvent evt)\r
97       {\r
98         if (getSize().width != width ||\r
99             getSize().height != sequencesHeight + graphHeight)\r
100         {\r
101           updateOverviewImage();\r
102         }\r
103       }\r
104     });\r
105 \r
106     addMouseMotionListener(this);\r
107 \r
108     addMouseListener(this);\r
109 \r
110     updateOverviewImage();\r
111 \r
112   }\r
113 \r
114 \r
115   public void mouseEntered(MouseEvent evt)\r
116   {}\r
117   public void mouseExited(MouseEvent evt)\r
118   {}\r
119   public void mouseClicked(MouseEvent evt)\r
120   {}\r
121   public void mouseMoved(MouseEvent evt)\r
122   {}\r
123   public void mousePressed(MouseEvent evt)\r
124   {\r
125     boxX = evt.getX();\r
126     boxY = evt.getY();\r
127 \r
128     checkValid();\r
129     repaint();\r
130   }\r
131 \r
132   public void mouseReleased(MouseEvent evt)\r
133   {\r
134     boxX = evt.getX();\r
135     boxY = evt.getY();\r
136     checkValid();\r
137     ap.setScrollValues( (int) (boxX / scalew / av.getCharWidth()),\r
138                          (int) (boxY / scaleh / av.getCharHeight()));\r
139   }\r
140 \r
141   public void mouseDragged(MouseEvent evt)\r
142   {\r
143     boxX = evt.getX();\r
144     boxY = evt.getY();\r
145     checkValid();\r
146     ap.setScrollValues( (int) (boxX / scalew / av.getCharWidth()),\r
147                          (int) (boxY / scaleh / av.getCharHeight()));\r
148 \r
149     repaint();\r
150     ap.repaint();\r
151   }\r
152 \r
153   void checkValid()\r
154   {\r
155     if (boxY < 0)\r
156     {\r
157       boxY = 0;\r
158     }\r
159 \r
160     if (boxY > sequencesHeight - boxHeight)\r
161     {\r
162       boxY = sequencesHeight - boxHeight + 1;\r
163     }\r
164 \r
165     if (boxX < 0)\r
166     {\r
167       boxX = 0;\r
168     }\r
169 \r
170     if (boxX > width - boxWidth)\r
171     {\r
172       boxX = width - boxWidth;\r
173     }\r
174   }\r
175 \r
176   /**\r
177    * DOCUMENT ME!\r
178    */\r
179   public void updateOverviewImage()\r
180   {\r
181     if (resizing)\r
182     {\r
183         resizeAgain = true;\r
184         return;\r
185     }\r
186 \r
187     if (av.showSequenceFeatures)\r
188    {\r
189      fr.featureGroups = ap.seqPanel.seqCanvas.getFeatureRenderer().featureGroups;\r
190      fr.featureColours = ap.seqPanel.seqCanvas.getFeatureRenderer().featureColours;\r
191      fr.sequenceFeatures = ap.seqPanel.seqCanvas.getFeatureRenderer().sequenceFeatures;\r
192    }\r
193 \r
194     resizing = true;\r
195 \r
196     if ( (getSize().width > 0) && (getSize().height > 0))\r
197     {\r
198       width = getSize().width;\r
199       sequencesHeight = getSize().height - graphHeight;\r
200     }\r
201     setSize(new Dimension(width, sequencesHeight + graphHeight));\r
202     setBoxPosition();\r
203 \r
204     Thread thread = new Thread(this);\r
205     thread.start();\r
206         repaint();\r
207   }\r
208 \r
209   // This is set true if the user resizes whilst\r
210   // the overview is being calculated\r
211     boolean resizeAgain = false;\r
212 \r
213   public void run()\r
214   {\r
215     miniMe = null;\r
216     int alwidth = av.alignment.getWidth();\r
217     int alheight = av.alignment.getHeight();\r
218 \r
219     if (av.showSequenceFeatures)\r
220     {\r
221       fr.renderOrder = ap.seqPanel.seqCanvas.getFeatureRenderer().renderOrder;\r
222       fr.featureGroups = ap.seqPanel.seqCanvas.getFeatureRenderer().featureGroups;\r
223       fr.featureColours = ap.seqPanel.seqCanvas.getFeatureRenderer().featureColours;\r
224       fr.sequenceFeatures = ap.seqPanel.seqCanvas.getFeatureRenderer().sequenceFeatures;\r
225     }\r
226 \r
227     if (getSize().width > 0 && getSize().height > 0)\r
228     {\r
229       width = getSize().width;\r
230       sequencesHeight = getSize().height - graphHeight;\r
231     }\r
232 \r
233     setSize(new Dimension(width, sequencesHeight + graphHeight));\r
234 \r
235     int fullsizeWidth = alwidth * av.getCharWidth();\r
236     int fullsizeHeight = alheight * av.getCharHeight();\r
237 \r
238     scalew = (float) width / (float) fullsizeWidth;\r
239     scaleh = (float) sequencesHeight / (float) fullsizeHeight;\r
240 \r
241     miniMe = nullFrame.createImage(width, sequencesHeight + graphHeight);\r
242 \r
243     Graphics mg = miniMe.getGraphics();\r
244     float sampleCol = (float) alwidth / (float) width;\r
245     float sampleRow = (float) alheight / (float) sequencesHeight;\r
246 \r
247     int lastcol=0, lastseq=0;\r
248     int xstart=0, ystart=0;\r
249     Color color = Color.yellow;\r
250     int col, sameRow = 0, sameCol = 0;\r
251     jalview.datamodel.SequenceI sequence;\r
252 \r
253     for (int row = 0; row <= sequencesHeight; row++)\r
254     {\r
255       if((int)(row*sampleRow)==lastseq)\r
256       {\r
257         sameRow ++;\r
258         continue;\r
259       }\r
260 \r
261       sequence = av.getAlignment().getSequenceAt(lastseq);\r
262 \r
263       for (col = 0; col < width; col++)\r
264       {\r
265         if((int)(col*sampleCol) == lastcol)\r
266         {\r
267           sameCol ++;\r
268           continue;\r
269         }\r
270 \r
271         lastcol = (int)(col*sampleCol);\r
272 \r
273         if(sequence.getLength()>lastcol)\r
274         {\r
275           color = sr.findSequenceColour(sequence, lastcol);\r
276 \r
277           if (av.showSequenceFeatures)\r
278             color = fr.findFeatureColour(color,\r
279                                         sequence,\r
280                                         lastcol);\r
281         }\r
282         else\r
283           color = color.white;\r
284 \r
285         mg.setColor(color);\r
286         if (sameCol == 1 && sameRow == 1)\r
287           mg.drawLine(xstart, ystart, xstart, ystart);\r
288         else\r
289           mg.fillRect(xstart, ystart, sameCol, sameRow);\r
290 \r
291         xstart = col;\r
292         sameCol = 1;\r
293       }\r
294 \r
295 \r
296       lastseq = (int)(row*sampleRow);\r
297       ystart = row;\r
298       sameRow = 1;\r
299     }\r
300 \r
301     if (av.conservation != null)\r
302     {\r
303       for (col = 0; col < width; col++)\r
304       {\r
305         lastcol = (int) (col * sampleCol);\r
306         {\r
307           mg.translate(col, sequencesHeight);\r
308           ap.annotationPanel.drawGraph(mg, av.conservation,\r
309                                        (int) (sampleCol) + 1,\r
310                                        graphHeight,\r
311                                        (int) (col * sampleCol),\r
312                                        (int) (col * sampleCol) + 1);\r
313           mg.translate( -col, -sequencesHeight);\r
314         }\r
315       }\r
316     }\r
317 \r
318 \r
319 \r
320     System.gc();\r
321 \r
322     resizing = false;\r
323 \r
324     setBoxPosition();\r
325 \r
326     if(resizeAgain)\r
327     {\r
328       resizeAgain = false;\r
329       updateOverviewImage();\r
330     }\r
331   }\r
332 \r
333   public void setBoxPosition()\r
334   {\r
335     int fullsizeWidth = av.alignment.getWidth() * av.getCharWidth();\r
336     int fullsizeHeight = av.alignment.getHeight() * av.getCharHeight();\r
337 \r
338     scalew = (float) width / (float) fullsizeWidth;\r
339     scaleh = (float) sequencesHeight / (float) fullsizeHeight;\r
340 \r
341     boxX = (int) (av.getStartRes() * av.getCharWidth() * scalew);\r
342     boxY = (int) (av.getStartSeq() * av.getCharHeight() * scaleh);\r
343     boxWidth = (int) ( (av.getEndRes() - av.getStartRes() + 1) *\r
344                       av.getCharWidth() * scalew);\r
345     boxHeight = (int) (av.getEndSeq() * av.getCharHeight() * scaleh) - boxY;\r
346     repaint();\r
347   }\r
348 \r
349   public void update(Graphics g)\r
350   {\r
351     paint(g);\r
352   }\r
353 \r
354   public void paint(Graphics g)\r
355   {\r
356     if (miniMe != null)\r
357     {\r
358       g.drawImage(miniMe, 0, 0, this);\r
359     }\r
360     else\r
361     {\r
362       g.setColor(Color.white);\r
363       g.fillRect(0, 0, getSize().width, getSize().height);\r
364       g.setColor(Color.black);\r
365       g.setFont(new Font("Verdana", Font.BOLD, 15));\r
366       g.drawString("Recalculating", 5, sequencesHeight / 2);\r
367       g.drawString("Overview.....", 5, (sequencesHeight / 2) + 20);\r
368     }\r
369 \r
370     g.setColor(Color.red);\r
371     g.drawRect(boxX, boxY, boxWidth, boxHeight);\r
372     g.drawRect(boxX + 1, boxY + 1, boxWidth - 2, boxHeight - 2);\r
373   }\r
374 \r
375 }\r