聽這個題目就挺奇怪——誰會在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