听这个题目就挺奇怪——谁会在awt/swing程序里加swt的东西呀,大多都是反过来,要做eclipse插件,却又想复用已有的awt做过的界面,不过想想要是以前用awt/swing做过的东西实在太大,改他们不容易呢?呵呵,我就接到这么个任务:
实验室里有一个所谓的软件体系结构建模工具,已经历时n年出了好多个版本,现在又要升级了,增加对运行时的管理,其中重要的一项就是配置底层的中间件,当然这个中间件也是我们实验室自己做的...算了,长话短说,就是要在原来的窗口里集成一个浏览器。swing提供的浏览器太弱了,去找开源的java浏览器插件也不容易,于是决定集成org.eclipse.swt.browser.Browser.虽然后来才知道eclipse里这帮鬼子们也偷懒,居然内部调用ie,算啦,凑合着用吧。
好,开始集成swt!
先配环境,去eclipse主页上下了一个swt的包,www.eclipse.ort/swt. 压缩包里有一个swt.jar,还有几个dll文件,添加jar包,然后运行,提示找不到swt-win32-3139,放了半天,终于在把这几个dll文件放到system32后,可以正常运行了。
然后无意中发现有一个叫SWT_AWT的类,很受挫,原来以为直接建一个Shell就可以用: (
SWT_AWT的思想简单说就是利用一个AWT里的Canvas建立一个Shell,然后就可以往这个Shell里添swt的东西了,至于canvas放在哪就无所谓了,可能在一个单独的窗口里,也可在某个大窗口的一部分中。我在网上找到了一段日本鬼子写的代码,可以说明这个类大概的用法:
JFrame frame = new JFrame();
Container cp = frame.getContentPane();
Canvas canvas = new Canvas();
cp.add(canvas,BorderLayout.CENTER);
frame.setVisible(true);
Display display = new Display();
Shell shell = SWT_AWT.new_Shell(display,canvas);
shell.setLayout(new FillLayout());
Button button = new Button(shell,SWT.PUSH);
button.setText("SWTのボタン");
shell.pack();
while (!shell.isDisposed()) {
if (!display.readAndDispatch()){
display.sleep ();
}
}
后面那个while循环很重要,有了这个循环,shell的Display才能不断地响应事件,没有事件时他会sleep等。但是问题也出在这个while上,他耗费你的当前线程,这时候你不能干别的工作了,原来的awt部分也不再响应事件了。
最直接的解决办法就是把这段代码放到一个单独的线程里,这是在eclipse的bug报告里找到的一段代码:
private class DisplayThread extends Thread {
private Display display;
public void run() {
display = Display.getDefault();
swtEventLoop();
}
private void swtEventLoop() {
while( true ) {
if (!display.readAndDispatch()) {
display.sleep();
}
}
}
public Display getDisplay() {
return display;
}
}
它还给了一个panel类,我在上面略加修改,这个类就是最后的包含了一个Shell的Panel.你可以一方面往他内部的shell中添各种swt的东西,也可以把整个类作为一个panel插到任何awt部分中。还有一个值得注意的是,这里用的是Panel,而不是JPanel,按eclipse文档的解释,“强烈建议使用一个 heavyweight component 作为根控件”。
public class SWTPane extends Panel {
DisplayThread displayThread;
private Canvas canvas;
public SWTPane() {
displayThread=new DisplayThread();
displayThread.start();
canvas = new Canvas();
setLayout( new BorderLayout() );
add( canvas, BorderLayout.CENTER );
}
public void addNotify() {
super.addNotify();
Display dis=displayThread.getDisplay();
dis.syncExec( new Runnable() {
public void run() {
Shell shell = SWT_AWT.new_Shell(displayThread.getDisplay(), canvas );
shell.setLayout( new FillLayout() );
final Browser browser = new Browser(shell, SWT.NONE);
browser.setLayoutData(BorderLayout.CENTER);
browser.setUrl("http://blog.csdn.net/fafey");
}
} );
}
}
其中dis.syncExec这个函数的意思是让display根据自己所在线程的情况,找一个合适的时机执行后面提供的代码。
但是这个代码也有问题,运行时在这个地方抛出NullPointerException
我觉得可能是因为在线程还没有建立好之前,先调用了getDisplay(),返回了一个null,没办法,在线程中增加同步机制吧!
public class DisplayThread extends Thread {
private Display display;
Object sem=new Object();
public void run() {
synchronized (sem){
display = Display.getDefault();
sem.notifyAll();
}
swtEventLoop();
}
private void swtEventLoop() {
while( true ) {
if (!display.readAndDispatch()) {
display.sleep();
}
}
}
public Display getDisplay() {
try{
synchronized (sem){
while(display==null)
sem.wait();
return display;
}
}
catch(Exception e){
return null;
}
}
}
这个类和上面那个合起来用就可以了。
转自:http://blog.csdn.net/fafey/archive/2006/05/10/721988.aspx