在awt/swing程序中添加swt (Browser控件的使用)【轉(zhuǎn)】
Posted on 2007-07-16 09:54 Matthew Chen 閱讀(2795) 評(píng)論(0) 編輯 收藏聽這個(gè)題目就挺奇怪——誰會(huì)在awt/swing程序里加swt的東西呀,大多都是反過來,要做eclipse插件,卻又想復(fù)用已有的awt做過的界面,不過想想要是以前用awt/swing做過的東西實(shí)在太大,改他們不容易呢?呵呵,我就接到這么個(gè)任務(wù):
實(shí)驗(yàn)室里有一個(gè)所謂的軟件體系結(jié)構(gòu)建模工具,已經(jīng)歷時(shí)n年出了好多個(gè)版本,現(xiàn)在又要升級(jí)了,增加對(duì)運(yùn)行時(shí)的管理,其中重要的一項(xiàng)就是配置底層的中間件,當(dāng)然這個(gè)中間件也是我們實(shí)驗(yàn)室自己做的...算了,長話短說,就是要在原來的窗口里集成一個(gè)瀏覽器。swing提供的瀏覽器太弱了,去找開源的java瀏覽器插件也不容易,于是決定集成org.eclipse.swt.browser.Browser.雖然后來才知道eclipse里這幫鬼子們也偷懶,居然內(nèi)部調(diào)用ie,算啦,湊合著用吧。
好,開始集成swt!
先配環(huán)境,去eclipse主頁上下了一個(gè)swt的包,www.eclipse.ort/swt. 壓縮包里有一個(gè)swt.jar,還有幾個(gè)dll文件,添加jar包,然后運(yùn)行,提示找不到swt-win32-3139,放了半天,終于在把這幾個(gè)dll文件放到system32后,可以正常運(yùn)行了。
然后無意中發(fā)現(xiàn)有一個(gè)叫SWT_AWT的類,很受挫,原來以為直接建一個(gè)Shell就可以用: (
SWT_AWT的思想簡(jiǎn)單說就是利用一個(gè)AWT里的Canvas建立一個(gè)Shell,然后就可以往這個(gè)Shell里添swt的東西了,至于canvas放在哪就無所謂了,可能在一個(gè)單獨(dú)的窗口里,也可在某個(gè)大窗口的一部分中。我在網(wǎng)上找到了一段日本鬼子寫的代碼,可以說明這個(gè)類大概的用法:
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 ();
}
}
后面那個(gè)while循環(huán)很重要,有了這個(gè)循環(huán),shell的Display才能不斷地響應(yīng)事件,沒有事件時(shí)他會(huì)sleep等。但是問題也出在這個(gè)while上,他耗費(fèi)你的當(dāng)前線程,這時(shí)候你不能干別的工作了,原來的awt部分也不再響應(yīng)事件了。
最直接的解決辦法就是把這段代碼放到一個(gè)單獨(dú)的線程里,這是在eclipse的bug報(bào)告里找到的一段代碼:
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;
}
}
它還給了一個(gè)panel類,我在上面略加修改,這個(gè)類就是最后的包含了一個(gè)Shell的Panel.你可以一方面往他內(nèi)部的shell中添各種swt的東西,也可以把整個(gè)類作為一個(gè)panel插到任何awt部分中。還有一個(gè)值得注意的是,這里用的是Panel,而不是JPanel,按eclipse文檔的解釋,“強(qiáng)烈建議使用一個(gè) 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這個(gè)函數(shù)的意思是讓display根據(jù)自己所在線程的情況,找一個(gè)合適的時(shí)機(jī)執(zhí)行后面提供的代碼。
但是這個(gè)代碼也有問題,運(yùn)行時(shí)在這個(gè)地方拋出NullPointerException
我覺得可能是因?yàn)樵诰€程還沒有建立好之前,先調(diào)用了getDisplay(),返回了一個(gè)null,沒辦法,在線程中增加同步機(jī)制吧!
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;
}
}
}
這個(gè)類和上面那個(gè)合起來用就可以了。
轉(zhuǎn)自:http://blog.csdn.net/fafey/archive/2006/05/10/721988.aspx