2003/03/13 02:12:17
[org.ibex.core.git] / src / org / xwt / plat / Java2.java
1 // Copyright 2002 Adam Megacz, see the COPYING file for licensing [GPL]
2 package org.xwt.plat;
3
4 import java.awt.*;
5 import java.awt.event.*;
6 import java.awt.image.*;
7 import java.awt.datatransfer.*;
8 import java.net.*;
9 import java.io.*;
10 import java.util.*;
11 import org.xwt.util.*;
12 import org.xwt.*;
13 import java.lang.reflect.*;
14
15
16 /** Platform class for most reasonable Java1.2+ Java2s */
17 public class Java2 extends AWT {
18
19     private boolean isJava14 = false;
20     protected boolean _supressDirtyOnResize() { return isJava14 ? false : true; }
21
22     public Java2() {
23         // disable the focus manager so we can intercept the tab key
24         String versionString = System.getProperty("java.version", "");
25         int secondDecimal = versionString.substring(versionString.indexOf('.') + 1).indexOf('.');
26         if (secondDecimal != -1) versionString = versionString.substring(0, secondDecimal);
27         double version = Double.parseDouble(versionString);
28         if (version >= 1.4) {
29             isJava14 = true;
30             try {
31                 Toolkit t = java.awt.Toolkit.getDefaultToolkit();
32                 Method m = java.awt.Toolkit.class.getMethod("setDynamicLayout", new Class[] { Boolean.class });
33                 m.invoke(t, new Object[] { Boolean.TRUE });
34             } catch (Exception e) {
35                 Log.log(this, "Exception while trying to enable AWT Dynamic Layout");
36                 Log.log(this, e);
37             }
38         }
39         javax.swing.FocusManager.setCurrentManager(new javax.swing.FocusManager() {
40                 public void processKeyEvent(Component focusedComponent, KeyEvent anEvent) { }
41                 public void focusPreviousComponent(Component aComponent) { }
42                 public void focusNextComponent(Component aComponent) { }
43             });
44     }
45
46     /** this is done with reflection in case a new version of the plugin comes out that doesn't let us pull the sun.plugin.* trick */
47     protected synchronized org.xwt.Proxy _detectProxy() {
48         return (org.xwt.Proxy)java.security.AccessController.doPrivileged(new java.security.PrivilegedAction() {
49                 public Object run() {
50                     try {
51                         org.xwt.Proxy pi = new org.xwt.Proxy();
52                         
53                         Class PluginProxyHandler = Class.forName("sun.plugin.protocol.PluginProxyHandler");
54                         Method getDefaultProxyHandler = PluginProxyHandler.getMethod("getDefaultProxyHandler", new Class[] { });
55                         Object proxyHandler = getDefaultProxyHandler.invoke(null, new Object[] { });
56                         
57                         Class ProxyHandler = Class.forName("sun.plugin.protocol.ProxyHandler");
58                         Method getProxyInfo = ProxyHandler.getMethod("getProxyInfo", new Class[] { URL.class });
59                         Object proxyInfo = getProxyInfo.invoke(proxyHandler, new Object[] { new URL("http://www.xwt.org") });
60                         
61                         Class ProxyInfo = Class.forName("sun.plugin.protocol.ProxyInfo");
62                         
63                         if (((Boolean)ProxyInfo.getMethod("isSocksUsed", new Class[] { }).invoke(proxyInfo, new Object[] { })).booleanValue()) {
64                             pi.socksProxyHost =
65                                 (String)ProxyInfo.getMethod("getSocksProxy", new Class[] { }).invoke(proxyInfo, new Object[] { });
66                             pi.socksProxyPort =
67                                 ((Integer)ProxyInfo.getMethod("getSocksPort", new Class[] { }).invoke(proxyInfo, new Object[] { })).intValue();
68                         }
69                         
70                         if (((Boolean)ProxyInfo.getMethod("isProxyUsed", new Class[] { }).invoke(proxyInfo, new Object[] { })).booleanValue()) {
71                             pi.httpProxyHost =
72                                 (String)ProxyInfo.getMethod("getProxy", new Class[] { }).invoke(proxyInfo, new Object[] { });
73                             pi.httpProxyPort =
74                                 ((Integer)ProxyInfo.getMethod("getPort", new Class[] { }).invoke(proxyInfo, new Object[] { })).intValue();
75                         }
76                         
77                         if (pi.httpProxyHost != null || pi.socksProxyHost != null) return pi;
78                         else return null;
79
80                     } catch (Throwable e) {
81                         if (Log.on) Log.log(this, "exception while querying sun.plugin.protocol.PluginProxyHandler: " + e);
82                         return null;
83                     }
84                 }});
85     }
86
87     protected Socket __getSocket(String host, int port, boolean ssl, boolean negotiate) throws IOException {
88         return super._getSocket(host, port, ssl, negotiate);
89     }
90     protected Socket _getSocket(final String host, final int port, final boolean ssl, final boolean negotiate) throws IOException {
91         return (Socket)java.security.AccessController.doPrivileged(new java.security.PrivilegedAction() {
92                 public Object run() {
93                     try {
94                         return __getSocket(host, port, ssl, negotiate);
95                     } catch (Exception e) {
96                         if (Log.on) Log.log(Java2.class, "Error attempting to create socket");
97                         if (Log.on) Log.log(Java2.class, e);
98                         return null;
99                     }
100                 }
101             });
102     }
103     
104     protected DoubleBuffer _createDoubleBuffer(int w, int h, Surface owner) { return new Java2DoubleBuffer(w, h); }
105     protected Surface _createSurface(final Box root, final boolean framed) {
106         return (Surface)java.security.AccessController.doPrivileged(new java.security.PrivilegedAction() {
107                 public Object run() { return isJava14 ? new Java14Surface(root, framed) : new Java2Surface(root, framed); }
108             });
109     }
110
111     // Inner Classes //////////////////////////////////////////////////////////////////
112
113     private static Cursor invisibleCursor =
114         Toolkit.getDefaultToolkit().createCustomCursor(new BufferedImage(2, 2, BufferedImage.TYPE_INT_ARGB),
115                                                        new Point(1, 1), "invisible");
116
117     protected static class Java2Surface extends AWTSurface {
118         
119         public Java2Surface(Box root, boolean framed) { super(root, framed); }
120
121         public void blit(DoubleBuffer s, int sx, int sy, int dx, int dy, int dx2, int dy2) {
122             if (ourGraphics == null) {
123                 ourGraphics = window.getGraphics();
124
125                 // sometimes jdk1.4 doesn't set the clip properly when we're in the middle of a resize
126                 ourGraphics.setClip(insets.left, insets.top, width + insets.left, height + insets.top);
127             }
128             _doDrawImage(ourGraphics, ((AWTDoubleBuffer)s).i, dx + insets.left, dy + insets.top, dx2 + insets.left, dy2 + insets.top,
129                          sx, sy, sx + (dx2 - dx), sy + (dy2 - dy), null);
130         }
131
132         protected void _setMinimized(boolean b) {
133             if (frame == null) {
134                 if (Log.on) Log.log(this, "JDK 1.2 can only minimize frames, not windows");
135                 return;
136             }
137             if (b) frame.setState(java.awt.Frame.ICONIFIED);
138             else frame.setState(java.awt.Frame.NORMAL);
139         }
140
141         public void syncCursor() {
142             if (cursor.equals("invisible")) window.setCursor(invisibleCursor);
143             else super.syncCursor();
144         }
145     }
146
147     protected static class Java14Surface extends Java2Surface implements WindowStateListener {
148         public Java14Surface(Box root, boolean framed) {
149             super(root, true);
150             // JDK1.4 doesn't like java.lang.Window's...
151             if (!framed) ((Frame)window).setUndecorated(true);
152             window.addWindowStateListener(this);
153             window.setVisible(true);
154         }
155
156         protected void makeVisible() { }
157         
158         protected void _setMaximized(boolean m) {
159             if (frame == null) {
160                 if (Log.on) Log.log(this, "JDK 1.4 can only maximize frames, not windows");
161                 return;
162             }
163             frame.setExtendedState(m ? Frame.MAXIMIZED_BOTH : (minimized ? Frame.ICONIFIED : Frame.NORMAL));
164         }
165         protected void _setMinimized(boolean m) {
166             if (frame == null) {
167                 if (Log.on) Log.log(this, "JDK 1.4 can only minimize frames, not windows");
168                 return;
169             }
170             frame.setExtendedState(m ? Frame.ICONIFIED : (maximized ? Frame.MAXIMIZED_BOTH : Frame.NORMAL));
171         }
172         public void windowStateChanged(WindowEvent e) {
173             if (e.getOldState() != e.getNewState()) {
174                 if ((e.getNewState() & Frame.MAXIMIZED_BOTH) != 0) Maximized(true);
175                 else if (((e.getOldState() & Frame.MAXIMIZED_BOTH) != 0) && (e.getNewState() & Frame.MAXIMIZED_BOTH) == 0)
176                     Maximized(false);
177             }
178         }
179     }
180
181     protected static class Java2DoubleBuffer extends AWTDoubleBuffer {
182         private static ColorModel cm = Toolkit.getDefaultToolkit().getColorModel();
183         private static Hashtable emptyHashtable = new Hashtable();
184         private static short[] sbank = null;
185         private static int[] ibank = null;
186         private static byte[] bbank = null;
187         private static int bank_start = 0;
188         
189         public void drawPicture(Picture source, int dx1, int dy1, int dx2, int dy2, int sx1, int sy1, int sx2, int sy2) {
190             _doDrawImage(g, ((AWTPicture)source).i, dx1, dy1, dx2, dy2, sx1, sy1, sx2, sy2, null);
191         }
192         
193         public Java2DoubleBuffer(int w, int h) {
194             SampleModel sm = cm.createCompatibleSampleModel(w, h);
195             int numSamples = w * h * sm.getNumDataElements();
196             DataBuffer buf = null;
197             if (sm.getDataType() == DataBuffer.TYPE_USHORT) {
198                 if (sbank == null || numSamples > 512 * 512 / 3) {
199                     buf = new DataBufferUShort(numSamples);
200                 } else {
201                     if (numSamples > sbank.length - bank_start) {
202                         bank_start = 0;
203                         sbank = new short[512 * 512];
204                     }
205                     buf = new DataBufferUShort(sbank, numSamples, bank_start);
206                     bank_start += numSamples;
207                 }
208             } else if (sm.getDataType() == DataBuffer.TYPE_BYTE) {
209                 if (bbank == null || numSamples > 512 * 512 / 3) {
210                     buf = new DataBufferByte(numSamples);
211                 } else {
212                     if (numSamples > bbank.length - bank_start) {
213                         bank_start = 0;
214                         bbank = new byte[512 * 512];
215                     }
216                     buf = new DataBufferByte(bbank, numSamples, bank_start);
217                     bank_start += numSamples;
218                 }
219             } else if (sm.getDataType() == DataBuffer.TYPE_INT) {
220                 if (ibank == null || numSamples > 512 * 512 / 3) {
221                     buf = new DataBufferInt(numSamples);
222                 } else {
223                     if (numSamples > ibank.length - bank_start) {
224                         bank_start = 0;
225                         ibank = new int[512 * 512];
226                     }
227                     buf = new DataBufferInt(ibank, numSamples, bank_start);
228                     bank_start += numSamples;
229                 }
230             }
231             i = new BufferedImage(cm, Raster.createWritableRaster(sm, buf, null), false,  emptyHashtable);
232             g = i.getGraphics();
233         }
234     }
235
236     /** used to avoid garbage creation with getClipBounds() */
237     private static Rectangle clipBounds = new Rectangle();
238     
239     protected static void _doDrawImage(Graphics g, Image i, int dx1, int dy1, int dx2, int dy2, int sx1, int sy1, int sx2, int sy2, ImageObserver o) {
240         if (dx1 == dx2 || dy1 == dy2) return;
241         g.drawImage(i, dx1, dy1, dx2, dy2, sx1, sy1, sx2, sy2, o);
242     }
243
244     protected org.xwt.Weak _getWeak(Object o) { return new Java2Weak(o); }
245     private static class Java2Weak extends java.lang.ref.WeakReference implements org.xwt.Weak {
246         public Java2Weak(Object o) { super(o); }
247     }
248
249     private String __getClipBoard() { return super._getClipBoard(); }
250     protected String _getClipBoard() {
251         return (String)java.security.AccessController.doPrivileged(new java.security.PrivilegedAction() {
252                 public Object run() { return __getClipBoard();  }
253             });
254     }
255
256     private void __setClipBoard(String s) { super._setClipBoard(s); }
257     protected void _setClipBoard(final String s) {
258         java.security.AccessController.doPrivileged(new java.security.PrivilegedAction() {
259                 public Object run() {
260                     __setClipBoard(s);
261                     return null;
262                 }
263             });
264     }
265
266     protected String getDescriptiveName() { return "Java 1.2+ JVM"; }
267
268     protected void _newBrowserWindow(String url) {
269         if (Main.applet == null) {
270             if (Log.on) Log.log(this, "Main.applet is null; cannot invoke showDocument()");
271             return;
272         }
273         if (Log.on) Log.log(this, "asking browser to show URL " + url);
274         try {
275             Main.applet.getAppletContext().showDocument(new URL(url), "_blank");
276         } catch (MalformedURLException e) {
277             if (Log.on) Log.log(this, e);
278         }
279     }
280
281 }