在SWING里嵌入SWT的组件

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了在SWING里嵌入SWT的组件相关的知识,希望对你有一定的参考价值。

http://zhanghan3366.blog.163.com/blog/static/6966305220110505351568/

——————————————————————————————————————————————————————————————————————

先配环境,去eclipse主页上下了一个swt的包,www.eclipse.ort/swt. 压缩包里有一个swt.jar,还有几个dll文件。

有 一个叫SWT_AWT的类, 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;
  }
 }

}

这个类和上面那个合起来用就可以了。

以上是关于在SWING里嵌入SWT的组件的主要内容,如果未能解决你的问题,请参考以下文章

如何在 Swing jframe 中嵌入 SWT 浏览器

SWT和Swing代码的区别

在 Java 中嵌入 Office

java如何内置visio控件

Java AWT/SWT/Swing:如何规划 GUI?

我在哪里可以买到商业 SWT 组件?