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