Picses' sky

Picses' sky
posts - 43, comments - 29, trackbacks - 0, articles - 24

听这个题目就挺奇怪——谁会在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

只有注册用户登录后才能发表评论。


网站导航: