Wrap alignment with annotations
[jalview.git] / src / jalview / io / HTMLOutput.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 package jalview.io;\r
20 \r
21 import java.io.*;\r
22 \r
23 import java.awt.*;\r
24 \r
25 import jalview.datamodel.*;\r
26 import jalview.gui.*;\r
27 import jalview.schemes.*;\r
28 \r
29 public class HTMLOutput\r
30 {\r
31   AlignViewport av;\r
32   SequenceRenderer sr;\r
33   Color color;\r
34 \r
35   public HTMLOutput(AlignViewport av)\r
36   {\r
37     this.av = av;\r
38     sr = new SequenceRenderer(av);\r
39 \r
40     JalviewFileChooser chooser = new JalviewFileChooser(jalview.bin.Cache.\r
41         getProperty(\r
42             "LAST_DIRECTORY"), new String[]\r
43         {"html"},\r
44         new String[]\r
45         {"HTML files"}, "HTML files");\r
46 \r
47     chooser.setFileView(new JalviewFileView());\r
48     chooser.setDialogTitle("Save as HTML");\r
49     chooser.setToolTipText("Save");\r
50 \r
51     int value = chooser.showSaveDialog(null);\r
52 \r
53     if (value == JalviewFileChooser.APPROVE_OPTION)\r
54     {\r
55       String choice = chooser.getSelectedFile().getPath();\r
56       jalview.bin.Cache.setProperty("LAST_DIRECTORY",\r
57                                     chooser.getSelectedFile().getParent());\r
58 \r
59       try\r
60       {\r
61         PrintWriter out = new java.io.PrintWriter(new java.io.FileWriter(\r
62             choice));\r
63         out.println("<HTML>");\r
64         out.println("<style type=\"text/css\">");\r
65         out.println("<!--");\r
66         out.print("td {font-family: \"" + av.getFont().getFamily() +\r
67                   "\", \"" + av.getFont().getName() + "\", mono; " +\r
68                   "font-size: " + av.getFont().getSize() + "px; ");\r
69 \r
70         if (av.getFont().getStyle() == Font.BOLD)\r
71         {\r
72           out.print("font-weight: BOLD; ");\r
73         }\r
74 \r
75         if (av.getFont().getStyle() == Font.ITALIC)\r
76         {\r
77           out.print("font-style: italic; ");\r
78         }\r
79 \r
80         out.println("text-align: center; }");\r
81 \r
82         out.println("-->");\r
83         out.println("</style>");\r
84         out.println("<BODY>");\r
85 \r
86         if (av.getWrapAlignment())\r
87         {\r
88           drawWrappedAlignment(out);\r
89         }\r
90         else\r
91         {\r
92           drawUnwrappedAlignment(out);\r
93         }\r
94 \r
95         out.println("\n</body>\n</html>");\r
96         out.close();\r
97         jalview.util.BrowserLauncher.openURL("file:///" + choice);\r
98       }\r
99       catch (Exception ex)\r
100       {\r
101         ex.printStackTrace();\r
102       }\r
103     }\r
104   }\r
105 \r
106   void drawUnwrappedAlignment(PrintWriter out)\r
107   {\r
108     out.println("<table border=\"1\"><tr><td>\n");\r
109     out.println(\r
110         "<table border=\"0\"  cellpadding=\"0\" cellspacing=\"0\">\n");\r
111 \r
112     //////////////\r
113     SequenceGroup[] groups;\r
114     SequenceI seq;\r
115     ColourSchemeI cs = null;\r
116     AlignmentI alignment = av.getAlignment();\r
117     String r;\r
118     String g;\r
119     String b;\r
120 \r
121     // draws the top row, the measure rule\r
122     out.println("<tr><td colspan=\"6\"></td>");\r
123 \r
124     int i = 0;\r
125 \r
126     for (i = 10; i < (alignment.getWidth() - 10); i += 10)\r
127     {\r
128       out.println("<td colspan=\"9\">" + i + "<br>|</td><td></td>");\r
129     }\r
130 \r
131     out.println("<td colspan=\"3\"></td><td colspan=\"3\">" + i +\r
132                 "<br>|</td>");\r
133     out.println("</tr>");\r
134 \r
135     for (i = 0; i < alignment.getHeight(); i++)\r
136     {\r
137       seq = alignment.getSequenceAt(i);\r
138       groups = alignment.findAllGroups(seq);\r
139 \r
140       if (av.getShowFullId())\r
141       {\r
142         out.println("<tr><td nowrap>" + seq.getDisplayId() +\r
143                     "&nbsp;&nbsp;</td>");\r
144       }\r
145       else\r
146       {\r
147         out.println("<tr><td nowrap>" + seq.getName() +\r
148                     "&nbsp;&nbsp;</td>");\r
149       }\r
150 \r
151       for (int res = 0; res < seq.getLength(); res++)\r
152       {\r
153         cs = av.getGlobalColourScheme();\r
154 \r
155         if (groups != null)\r
156         {\r
157           for (int k = 0; k < groups.length; k++)\r
158           {\r
159             if ( (groups[k].getStartRes() <= res) &&\r
160                 (groups[k].getEndRes() >= res))\r
161             {\r
162               cs = groups[k].cs;\r
163 \r
164               break;\r
165             }\r
166           }\r
167         }\r
168 \r
169         color = sr.getResidueBoxColour(cs, seq, res);\r
170 \r
171         if (color.getRGB() < -1)\r
172         {\r
173           out.println("<td bgcolor=\"#" +\r
174                       jalview.util.Format.getHexString(color) + "\">" +\r
175                       seq.getCharAt(res) + "</td>");\r
176         }\r
177         else\r
178         {\r
179           out.println("<td>" + seq.getCharAt(res) + "</td>");\r
180         }\r
181       }\r
182 \r
183       out.println("</tr>");\r
184     }\r
185 \r
186     //////////////\r
187     out.println("</table>");\r
188     out.println("</td></tr></table>");\r
189   }\r
190 \r
191   void drawWrappedAlignment(PrintWriter out)\r
192   {\r
193     ////////////////////////////////////\r
194     /// How many sequences and residues can we fit on a printable page?\r
195     AlignmentI al = av.getAlignment();\r
196     SequenceGroup[] groups;\r
197     SequenceI seq;\r
198     ColourSchemeI cs = null;\r
199     String r;\r
200     String g;\r
201     String b;\r
202 \r
203     out.println("<table border=\"1\"><tr><td>\n");\r
204     out.println(\r
205         "<table border=\"0\"  cellpadding=\"0\" cellspacing=\"0\">\n");\r
206 \r
207     for (int startRes = 0; startRes < al.getWidth();\r
208          startRes += av.getWrappedWidth())\r
209     {\r
210       int endRes = startRes + av.getWrappedWidth();\r
211 \r
212       if (endRes > al.getWidth())\r
213       {\r
214         endRes = al.getWidth();\r
215       }\r
216 \r
217       out.println("<tr>");\r
218       out.println("<td colspan=\"6\">&nbsp;</td>");\r
219 \r
220       for (int i = startRes + 10; i < endRes; i += 10)\r
221       {\r
222         out.println("<td colspan=\"9\">" + i + "<br>|</td><td></td>");\r
223       }\r
224 \r
225       // out.println("<td colspan=\"3\"></td><td colspan=\"3\">" + i +\r
226       //             "<br>|</td>");\r
227       out.println("</tr>");\r
228 \r
229       for (int s = 0; s < al.getHeight(); s++)\r
230       {\r
231         out.println("<tr>");\r
232         seq = al.getSequenceAt(s);\r
233         groups = al.findAllGroups(seq);\r
234 \r
235         if (av.getShowFullId())\r
236         {\r
237           out.println("<td nowrap>" + seq.getDisplayId() +\r
238                       "&nbsp;&nbsp;</td>");\r
239         }\r
240         else\r
241         {\r
242           out.println("<td nowrap>" + seq.getName() +\r
243                       "&nbsp;&nbsp;</td>");\r
244         }\r
245 \r
246         for (int res = startRes; res < endRes; res++)\r
247         {\r
248           cs = av.getGlobalColourScheme();\r
249 \r
250           if (groups != null)\r
251           {\r
252             for (int k = 0; k < groups.length; k++)\r
253             {\r
254               if ( (groups[k].getStartRes() <= res) &&\r
255                   (groups[k].getEndRes() >= res))\r
256               {\r
257                 cs = groups[k].cs;\r
258 \r
259                 break;\r
260               }\r
261             }\r
262           }\r
263 \r
264           color = sr.getResidueBoxColour(cs, seq, res);\r
265 \r
266           if (color.getRGB() < -1)\r
267           {\r
268             r = Integer.toHexString(color.getRed());\r
269 \r
270             if (r.length() < 2)\r
271             {\r
272               r = "0" + r;\r
273             }\r
274 \r
275             g = Integer.toHexString(color.getGreen());\r
276 \r
277             if (g.length() < 2)\r
278             {\r
279               g = "0" + g;\r
280             }\r
281 \r
282             b = Integer.toHexString(color.getBlue());\r
283 \r
284             if (b.length() < 2)\r
285             {\r
286               b = "0" + b;\r
287             }\r
288 \r
289             out.println("<td bgcolor=\"#" + r + g + b + "\">" +\r
290                         seq.getCharAt(res) + "</td>");\r
291           }\r
292           else\r
293           {\r
294             out.println("<td>" + seq.getCharAt(res) + "</td>");\r
295           }\r
296         }\r
297 \r
298         out.println("</tr>");\r
299       }\r
300 \r
301       if (endRes < al.getWidth())\r
302       {\r
303         out.println("<tr><td height=\"5\"></td></tr>");\r
304       }\r
305     }\r
306 \r
307     out.println("</table>");\r
308     out.println("</table>");\r
309   }\r
310 \r
311   public static String getImageMapHTML()\r
312   {\r
313     return new String(\r
314       "<html>\n"\r
315       +"<head>\n"\r
316       +"<script language=\"JavaScript\">\n"\r
317       +"var ns4 = document.layers;\n"\r
318       +"var ns6 = document.getElementById && !document.all;\n"\r
319       +"var ie4 = document.all;\n"\r
320       +"offsetX = 0;\n"\r
321       +"offsetY = 20;\n"\r
322       +"var toolTipSTYLE=\"\";\n"\r
323       +"function initToolTips()\n"\r
324       +"{\n"\r
325       +"  if(ns4||ns6||ie4)\n"\r
326       +"  {\n"\r
327       +"    if(ns4) toolTipSTYLE = document.toolTipLayer;\n"\r
328       +"    else if(ns6) toolTipSTYLE = document.getElementById(\"toolTipLayer\").style;\n"\r
329       +"    else if(ie4) toolTipSTYLE = document.all.toolTipLayer.style;\n"\r
330       +"    if(ns4) document.captureEvents(Event.MOUSEMOVE);\n"\r
331       +"    else\n"\r
332       +"    {\n"\r
333       +"      toolTipSTYLE.visibility = \"visible\";\n"\r
334       +"      toolTipSTYLE.display = \"none\";\n"\r
335       +"    }\n"\r
336       +"    document.onmousemove = moveToMouseLoc;\n"\r
337       +"  }\n"\r
338       +"}\n"\r
339       +"function toolTip(msg, fg, bg)\n"\r
340       +"{\n"\r
341       +"  if(toolTip.arguments.length < 1) // hide\n"\r
342       +"  {\n"\r
343       +"    if(ns4) toolTipSTYLE.visibility = \"hidden\";\n"\r
344       +"    else toolTipSTYLE.display = \"none\";\n"\r
345       +"  }\n"\r
346       +"  else // show\n"\r
347       +"  {\n"\r
348       +"    if(!fg) fg = \"#555555\";\n"\r
349       +"    if(!bg) bg = \"#FFFFFF\";\n"\r
350       +"    var content =\n"\r
351       +"    '<table border=\"0\" cellspacing=\"0\" cellpadding=\"1\" bgcolor=\"' + fg + '\"><td>' +\n"\r
352       +"    '<table border=\"0\" cellspacing=\"0\" cellpadding=\"1\" bgcolor=\"' + bg + \n"\r
353       +"    '\"><td align=\"center\"><font face=\"sans-serif\" color=\"' + fg +\n"\r
354       +"    '\" size=\"-2\">&nbsp;' + msg +\n"\r
355       +"    '&nbsp;</font></td></table></td></table>';\n"\r
356       +"    if(ns4)\n"\r
357       +"    {\n"\r
358       +"      toolTipSTYLE.document.write(content);\n"\r
359       +"      toolTipSTYLE.document.close();\n"\r
360       +"      toolTipSTYLE.visibility = \"visible\";\n"\r
361       +"    }\n"\r
362       +"    if(ns6)\n"\r
363       +"    {\n"\r
364       +"      document.getElementById(\"toolTipLayer\").innerHTML = content;\n"\r
365       +"      toolTipSTYLE.display='block'\n"\r
366       +"    }\n"\r
367       +"    if(ie4)\n"\r
368       +"    {\n"\r
369       +"      document.all(\"toolTipLayer\").innerHTML=content;\n"\r
370       +"      toolTipSTYLE.display='block'\n"\r
371       +"    }\n"\r
372       +"  }\n"\r
373       +"}\n"\r
374       +"function moveToMouseLoc(e)\n"\r
375       +"{\n"\r
376       +"  if(ns4||ns6)\n"\r
377       +"  {\n"\r
378       +"    x = e.pageX;\n"\r
379       +"    y = e.pageY;\n"\r
380       +"  }\n"\r
381       +"  else\n"\r
382       +"  {\n"\r
383       +"    x = event.x + document.body.scrollLeft;\n"\r
384       +"    y = event.y + document.body.scrollTop;\n"\r
385       +"  }\n"\r
386       +"  toolTipSTYLE.left = x + offsetX;\n"\r
387       +"  toolTipSTYLE.top = y + offsetY;\n"\r
388       +"  return true;\n"\r
389       +"}\n"\r
390       +"</script>\n"\r
391       +"</head>\n"\r
392       +"<body>\n"\r
393       +"<div id=\"toolTipLayer\" style=\"position:absolute; visibility: hidden\"></div>\n"\r
394       +"<script language=\"JavaScript\"><!--\n"\r
395       +"initToolTips(); //--></script>\n");\r
396 \r
397   }\r
398 }\r