GTK,作为与 QT并存的另外一套图形库,深受C程序员的喜爱。在同一程序中使用 QT 和GTK,甚至是其它图形库,对于编写嵌入式(Embeded)程序是十分有意义的。这也正式本文章所阐述的主题。
嵌入式程序的概念嵌入式程序一般没有自己的主循环体,它直接使用窗口的ID来嵌入到其它窗口中。在Xlib 层次上,使用函数 XReparentWindow 重新设置一个窗口的父窗口:
1
2
| XReparentWindow(Display *display, Window w,
Window parent, int x, int y)
|
即可以把窗口 w 嵌入到窗口 parent 中。
嵌入式程序一般使用动态库载入的方法,这样使用任何X窗口系统下的图形库,都可以通过输出(Export)其窗口ID的方法和其它程序库协同工作。嵌入程序需要父窗口来管理窗口的一些操作,如窗口的尺寸管理,键盘聚焦等等。一个完整的应用程序也可以嵌入到其它窗口中,一般地,建立主窗口时使窗口不受管理,然后使用XReparentWindow()设置父窗口。
在 QT 中使用 Motif在 QT 2.x 版本中有一个 QT 的 Xt/Motif的扩展,它可以使用户编写同时支持 Xt/Motif 和 Qt 的程序。QT对于Xt/Motif 的扩展是建立了类 QXtWidget 和类QXtApplication,它主要处理了嵌入窗口的底层事件。
建立一个以 QT为主的包含Motif组件(Widget)的程序,只要使用QXtApplication建立主程序,设置一个可以包含QT和Motif窗口的主窗口即可,例如:
1
2
3
4
5
| QXtApplication app(argc, argv, "TwoEditors");
TwoEditors m;
app.setMainWidget(&m);
m.show();
return app.exec();
|
在 m 中设置两个窗口,一个是QT类建立的,一个是QXtWidget类建立的,
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
| class TwoEditors : public QMainWindow {
QMultiLineEdit* qtchild;
EncapsulatedXtWidget* xtchild;
public:
TwoEditors() : QMainWindow( 0, "mainWindow") {
QPopupMenu* file = new QPopupMenu( this );
file->insertItem("E&xit", qApp, SLOT( quit() ) );
menuBar()->insertItem( "&File", file );
statusBar();
QSplitter* splitter = new QSplitter( this );
splitter->setOpaqueResize( TRUE );
setCentralWidget( splitter );
xtchild = new EncapsulatedXtWidget( splitter );
qtchild = new QMultiLineEdit( splitter );
qtchild->setText(QTEDMSG);
}
};
|
在 QT 中使用 GTK在 QT 中使用 GTK 稍复杂一些,主要是因为必须把 QT 的主循环和 GTK的主循环融合到一起比较复杂。在软件包 QGtkWidget 中,作者使用了 GTK的一部分代码,QT的一部分代码和KDE的一部分代码,构造了 QGtkWidget类和 QGtkApplication类。
编写程序的结构和上面的例子类似,
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
| QGtkApplication a( argc, argv );
QMainWindow *wnd=new MainWnd();
QPushButton *Qt=new QPushButton("This is Qt!", wnd);
QGtkWidget *Gtk=new QGtkWidget(wnd);
wnd->resize(200,100);
Qt->setGeometry(0,0,100,50);
Gtk->setGeometry(100,0,100,100);
GtkWidget *box=gtk_vbox_new(FALSE, 0);
gtk_container_add(GTK_CONTAINER(Gtk->getGtkWidget()), box);
GtkWidget *b=gtk_button_new_with_label("This is GTK!");
gtk_box_pack_start(GTK_BOX(box), b, TRUE, TRUE, 0);
gtk_widget_show(b);
gtk_widget_show(box);
wnd->show();
a.setMainWidget( wnd );
gtk_widget_show(b);
gtk_widget_show(box);
return a.exec();
|
|