wxWidgets是一個(gè)跨平臺(tái)的軟件開(kāi)發(fā)包。它誕生于1992年,最初的名子是wxWindows,但由于Microsoft的抗議,在2004年改名為wxWidgets。它最初是被設(shè)計(jì)成跨平臺(tái)的GUI軟件開(kāi)發(fā)包,但后來(lái)隨著越來(lái)越多的人參與進(jìn)來(lái),為wxWidgets加入了許多非GUI的功能,如多線程(MultiThread)、網(wǎng)絡(luò)(Network)等。并且從最初的只支持C++語(yǔ)言,逐漸發(fā)展成為支持?jǐn)?shù)種語(yǔ)言(如Python、Perl、C#、Basic等)。因此,現(xiàn)在的wxWidgets已經(jīng)不再是單純的跨平臺(tái)的GUI軟件開(kāi)發(fā)包,而是一個(gè)可以支持多種操作系統(tǒng)平臺(tái)的能夠在多種語(yǔ)言中使用的通用跨平臺(tái)軟件開(kāi)發(fā)包。
目前支持C++的軟件開(kāi)發(fā)包非常多,比較有名的除了wxWidgets外,還有一些其它的軟件開(kāi)發(fā)包,如MFC、QT、ACE等。即然有這么多開(kāi)發(fā)包,那么我們?yōu)槭裁匆褂脀xWidgets呢?在給出答案之前,讓我們首先來(lái)看一看上述的三種軟件開(kāi)發(fā)包的特性。
MFC是Microsoft提供的軟件開(kāi)發(fā)包。MFC雖然十分強(qiáng)大,但它只能運(yùn)行在Windows下運(yùn)行。而且它是收費(fèi)的。
QT是由Trolltech 公司開(kāi)發(fā)的一套跨平臺(tái)軟件開(kāi)發(fā)包。它和wxWidgets類似,但是QT只在linux下免費(fèi),而在Windows或Unix下使用QT要向Trolltech公司支付版權(quán)費(fèi)。
注:其中免費(fèi)中的“是/否”代表QT在linux平臺(tái)上的Free
Edition是免費(fèi)的,而在windows和unix下使用QT是收費(fèi)的。而開(kāi)源中的“是/否”代表QT有一個(gè)基于GPL的開(kāi)源版本,但要進(jìn)行商業(yè)開(kāi)發(fā),需要使用它的商業(yè)版本。
使用 wxWidgets 編寫(xiě)程序
學(xué)習(xí)一種編程語(yǔ)言的最好方法就是用它去編寫(xiě)程序,學(xué)習(xí)wxWidgets也不例外。由于wxWidgets的主要功能是實(shí)現(xiàn)跨平臺(tái)的GUI,因此,本文主要從GUI入手,討論wxWidgets在C++中如何編寫(xiě)跨平臺(tái)的應(yīng)用程序
1. 應(yīng)用程序類的建立
使用wxWidgets建立系統(tǒng)需要一個(gè)類來(lái)描述整個(gè)應(yīng)用程序。這個(gè)類必須從wxApp類繼承。
class MyApp : public wxApp //應(yīng)用程序類
{
public:
virtual bool OnInit(); // 在應(yīng)用程序啟動(dòng)時(shí)調(diào)用,如果返回false,退出應(yīng)用程序
};
這個(gè)類只覆蓋了wxApp的一個(gè)虛方法OnInit。可以用這個(gè)方法在程序啟動(dòng)時(shí)做一些驗(yàn)證,如果驗(yàn)證失敗,可以通過(guò)返回false退出應(yīng)用程序。當(dāng)然,由于這個(gè)函數(shù)是應(yīng)用程序的入口點(diǎn),所以建立主窗體的工作要在這個(gè)函數(shù)中完成。
2. 建立窗體類
wxWidgets中關(guān)于窗體的類很多,如果要建立一般窗體的話,可以從wxFrame繼承。
class MyFrame : public wxFrame //窗體類
{
public:
MyFrame(const wxString& title); // 窗體的構(gòu)造函數(shù)
};
3. 向窗體中加入控件
在本文中向這個(gè)窗體加入了一個(gè)菜單條(Menu Bar)、一個(gè)狀態(tài)條、一個(gè)Panel和一個(gè)按鈕。一般我們會(huì)在主窗體的構(gòu)造函數(shù)中加入這些控件。
MyFrame::MyFrame(const wxString& title) : wxFrame(NULL, wxID_ANY, title)
{
wxMenu *fileMenu = new wxMenu; // 建立“文件”菜單
wxMenu *helpMenu = new wxMenu; // 建立“幫助”菜單
// 向菜單中添加子項(xiàng)
helpMenu->Append(wxID_ABOUT, _T("關(guān)于

\tF1"), _T("顯示關(guān)于對(duì)話框"));
fileMenu->Append(wxID_EXIT, _T("退出\tAlt-X"), _T("退出應(yīng)用程序"));
wxMenuBar *menuBar = new wxMenuBar(); // 建立一個(gè)菜單條
menuBar->Append(fileMenu, _T("文件")); //將“文件”菜單加入到菜單條
menuBar->Append(helpMenu, _T("幫助")); //將“幫助”菜單加入到菜單條
SetMenuBar(menuBar); //將菜單條放到窗體上
wxPanel *panel = new wxPanel(this); //建立一個(gè)Panel
wxButton *button = new wxButton(panel, wxID_ABOUT, "關(guān)于", wxPoint(20, 20), wxSize(50, 30)); //建立一個(gè)Button
CreateStatusBar(2); //建立一個(gè)兩欄的狀態(tài)欄
SetStatusText(_T("歡迎使用wxWidgets!")); //設(shè)置狀態(tài)欄的文本
在數(shù)組sample_xpm中描述了sample.ico的屬性和圖標(biāo)本身。如X代表紅色; o代表黃色等。然后在源程序中通過(guò)include
“sample.xpm”引用這個(gè)資源文件。要想從這個(gè)資源文件中裝載圖標(biāo)。可使用SetIcon(wxICON(sample)); wxICON讀取資源文件,而SetIcon將這個(gè)圖標(biāo)設(shè)置為frame的標(biāo)題欄圖標(biāo)。要想將ico文件轉(zhuǎn)換為這種資源文件,可使用一個(gè)免費(fèi)軟件XnView進(jìn)行轉(zhuǎn)換。
5. 顯示主窗體
顯示主窗體非常簡(jiǎn)單,只需要將上面建立的MyFrame類實(shí)例化,并調(diào)用wxFrame的Show方法顯示即可。這些代碼可以寫(xiě)在MyApp類的OnInit方法中。
bool MyApp::OnInit()
{
//建立MyFrame類的實(shí)例
MyFrame *frame = new MyFrame(_T("第一個(gè)wxWidgets程序"));
frame->Show(true); //顯示主窗體
return true; //必須返回true,否則應(yīng)用程序?qū)⑼顺?br>
}
在以上代碼中Show方法有一個(gè)參數(shù),如果為true,則以模式窗口的形式顯示,否則以非模式窗口的形式顯示。
6. 向窗體中加入事件
到目前為止,這個(gè)程序的界面已經(jīng)完成了,但還未響應(yīng)任何事件,下面就詳細(xì)闡述如何向這個(gè)應(yīng)用程序中加入事件代碼。
對(duì)于事件來(lái)說(shuō),一般都會(huì)由兩部分組成。
(1)調(diào)用事件部分
當(dāng)程序發(fā)生某個(gè)動(dòng)作時(shí),如點(diǎn)擊按鈕;選中某個(gè)控件,可能需要執(zhí)行一段代碼。而這段代碼一般是由系統(tǒng)負(fù)責(zé)調(diào)用的,也就是說(shuō)系統(tǒng)通過(guò)事件函數(shù)指針調(diào)用相應(yīng)的代碼。
(2)事件函數(shù)本身
事件函數(shù)與普通函數(shù)一樣,只不過(guò)它是在發(fā)生了事件之后,由系統(tǒng)調(diào)用的。
在wxWidgets中是通過(guò)事件哈希表(Event Hash Table)來(lái)進(jìn)行事件處理的,即將相應(yīng)的事件函數(shù)指針保存在一個(gè)哈希表中,然后當(dāng)事件發(fā)生時(shí),從這個(gè)哈希表中找到相應(yīng)的事件函數(shù)指針,然后通過(guò)函數(shù)指針調(diào)用函數(shù)。在使用事件哈希表之前,必須定義它。由于定義哈希表非常復(fù)雜,而且每個(gè)需要處理事件的類都需要同樣的代碼,因此,wxWidgets為此定義了一個(gè)宏DECLARE_EVENT_TABLE()來(lái)定義哈希表。可將這個(gè)宏寫(xiě)在MyFrame類的任何位置。它相當(dāng)于將以下語(yǔ)句放到了MyFrame類中。
private:
static const wxEventTableEntry sm_eventTableEntries[];
protected:
static const wxEventTable sm_eventTable;
virtual const wxEventTable* GetEventTable() const;
static wxEventHashTable sm_eventHashTable;
virtual wxEventHashTable& GetEventHashTable() const;
其中靜態(tài)數(shù)組變量sm_eventTableEntries保存了MyFrame類中的所有的事件信息。
上面的代碼聲明了處理事件哈希表的一些方法,即然聲明了,就得實(shí)現(xiàn)。由于實(shí)現(xiàn)代碼也都一樣,因此,wxWidgets也為實(shí)現(xiàn)這些方法定義了一組宏。實(shí)現(xiàn)這些方法的宏如下所示。
BEGIN_EVENT_TABLE(MyFrame, wxFrame)
EVT_MENU(wxID_EXIT, MyFrame::OnQuit)
EVT_MENU(wxID_ABOUT, MyFrame::OnAbout)
EVT_BUTTON(wxID_ABOUT, MyFrame::OnAbout)
END_EVENT_TABLE()
其中BEGIN_EVENT_TABLE(…)實(shí)現(xiàn)了上面定義的方法,以及初始化了靜態(tài)變量sm_eventTable。后面兩個(gè)EVT_MENU和一個(gè)EVT_BUTTON宏初始化了靜態(tài)變量sm_eventTableEntries,即將這兩個(gè)事件函數(shù)的指針(button和about菜單使用一個(gè)事件函數(shù)OnAbout)和控件ID保存在sm_eventTableEntries中,最后的END_EVENT_TABLE()宏做為一個(gè)空的事件函數(shù)指針賦給了sm_eventTableEntries,這有些象C語(yǔ)言中處理字符串,將最后一個(gè)字符賦為’"0’,這樣就可以知道哪是結(jié)尾了。
向窗體中加入事件的最后一步是聲明和實(shí)現(xiàn)事件函數(shù)。在本例中聲明了兩個(gè)事件函數(shù)。
void OnQuit(wxCommandEvent& event);
void OnAbout(wxCommandEvent& event);
可以將這兩個(gè)函數(shù)聲明放到MyFrame中的任何位置。下面是它們的實(shí)現(xiàn)代碼。
void MyFrame::OnQuit(wxCommandEvent& WXUNUSED(event))
{
Close(true);
}
void MyFrame::OnAbout(wxCommandEvent& WXUNUSED(event))
{
wxString msg;
msg.Printf( _T("這是一個(gè)關(guān)于對(duì)話框的例子.\n")
_T("歡迎使用 %s"), wxVERSION_STRING);
wxMessageBox(msg, _T("¹關(guān)于"), wxOK | wxICON_INFORMATION, this);
}
其中OnQuit函數(shù)調(diào)用Close(true)關(guān)閉MyFrame,由于MyFrame是主窗體,因此,在MyFrame關(guān)閉后,應(yīng)用程序也隨之關(guān)閉了。OnAbout使用wxMessageBox函數(shù)彈出一個(gè)信息對(duì)話框。
7. 運(yùn)行程序
到目前為止,這個(gè)程序的代碼已經(jīng)基本完成了,但是在前面曾說(shuō)過(guò),MyApp中的OnInit方法在應(yīng)用程序啟動(dòng)時(shí)執(zhí)行,那么是誰(shuí)調(diào)用了OnInit方法呢?答案當(dāng)然是wxWidgets。wxWidgets為了調(diào)用這個(gè)方法,提供了一個(gè)宏IMPLEMENT_APP(…),這個(gè)宏有一個(gè)參數(shù),需要將MyApp做為參數(shù)傳入。即IMPLEMENT_APP(MyApp)。這個(gè)宏相當(dāng)于一個(gè)WinMain函數(shù)(和控制臺(tái)程序的main函數(shù)類似),即在WinMain函數(shù)中調(diào)用了MyApp中的OnInit函數(shù)。在加入這個(gè)宏后,就可使用一個(gè)C++編譯器將以上的源程序編譯生成exe文件了。應(yīng)用程序的界面如圖1、圖2所示。
圖1 Windows下的程序界面
圖2 Linux下的程序界面
注:在windows下使用的wxWidgets版本是wxWidgets2.6.2,在linux下使用的wxWidgets版本號(hào)是wxWidgets2.6.3,因此,在windows和linux下的wxVERSION_STRING值不一樣。
wxWidgets 的優(yōu)勢(shì)和不足
通過(guò)上面的介紹,相信讀者已經(jīng)對(duì)如何使用wxWidgets編寫(xiě)GUI程序有了一定的了解。wxWidgets在開(kāi)發(fā)跨平臺(tái)的軟件上有著許多其它軟件開(kāi)發(fā)包不具備的優(yōu)勢(shì),下面就總結(jié)一下wxWidgets所具有的優(yōu)勢(shì)。
1. 跨平臺(tái)
wxWidgets支持非常多的操作系統(tǒng)平臺(tái),如Windows、Linux、Unix等。
2. 豐富的組件
wxWidgets擁有上百個(gè)組件可供用戶選擇。有了這些組件,將會(huì)給我們帶來(lái)更加豐富的用戶體驗(yàn)。
3. 支持多種語(yǔ)言
wxWidgets不僅可以在C++中使用,而且也可以在其它語(yǔ)言中使用,這些語(yǔ)言包括python、perl、c#等。
4. 使用本地控制
從上面給出的兩個(gè)應(yīng)用程序界面可以看出,在Windows和Linux下運(yùn)行這個(gè)應(yīng)用程序保持了各自的風(fēng)格。這是因?yàn)閣xWidgets采用了本地的API,而不象其它的跨平臺(tái)庫(kù)去模擬它們。因此,使用wxWidgets開(kāi)發(fā)和在Windows下使用Win32
API或在Linux下使用GTK開(kāi)發(fā)沒(méi)有什么區(qū)別。
5. 免費(fèi)開(kāi)源
這個(gè)世界上免費(fèi)的開(kāi)發(fā)包很多,強(qiáng)大的開(kāi)發(fā)包也很多,當(dāng)然,開(kāi)源的開(kāi)發(fā)包就更多了。但是要想同時(shí)滿足這三點(diǎn):免費(fèi)、開(kāi)源、強(qiáng)大,又同時(shí)具有本地程序一樣的性能,恐怕wxWidgets是唯一的選擇,至少是最佳的選擇。
相信上面關(guān)于wxWidgets的5個(gè)優(yōu)勢(shì)已經(jīng)足以成為我們選擇它的理由了。也就是說(shuō),如果選擇wxWidgets,不僅可以獲得強(qiáng)大的功能、卓越的性能,而且您不必為此付一分錢。當(dāng)然,人無(wú)完人、物無(wú)完物。wxWidgets也并不是沒(méi)有缺點(diǎn)。下面就說(shuō)一下wxWidgets的不足之處。
1. IDE支持不夠
對(duì)于wxWidgets來(lái)說(shuō),最大的優(yōu)點(diǎn)也就是它最大的缺點(diǎn)。由于wxWidgets所提供的組件很多,但到現(xiàn)在為止還沒(méi)有一個(gè)強(qiáng)大的IDE來(lái)支持它,這將給大型系統(tǒng)的開(kāi)發(fā)帶來(lái)麻煩。
2. 對(duì)雙字節(jié)字符的支持不理想
wxWidgets中的有些組件,如xml組件,無(wú)法識(shí)別雙字節(jié)字符,如漢字會(huì)被認(rèn)為是非法字符而無(wú)法裝載xml文檔。
綜合上述,wxWidgets從總體上來(lái)說(shuō)還是一個(gè)非常強(qiáng)大的跨平臺(tái)軟件開(kāi)發(fā)包。如果您沒(méi)有足夠的資金來(lái)購(gòu)買商業(yè)的軟件開(kāi)發(fā)包,也許wxWidgets是最好的選擇。雖然wxWidgets也有一些不足,但這并不能阻礙wxWidgets的發(fā)展。wxWidgets的功能還很多,由于篇幅所限,本文只能從一個(gè)簡(jiǎn)單的例子來(lái)討論如何用wxWidgets來(lái)開(kāi)發(fā)一個(gè)跨平臺(tái)的GUI程序,如果讀者對(duì)wxWidgets感性趣,可以訪問(wèn)http://www.wxWidgets.org獲得更多的信息。
要在Linux下使用Eclipse開(kāi)發(fā)wxWidgets程序,請(qǐng)讀者參閱
《快速配置Linux + Eclipse + wxWidgets開(kāi)發(fā)環(huán)境》