随机产生的三个点,怎么画出一条弧线?(100分)

yanfengqiang 2003-08-29 08:14:27
用什么方法?

谢谢大家啦。
...全文
253 17 打赏 收藏 转发到动态 举报
写回复
用AI写文章
17 条回复
切换为时间正序
请发表友善的回复…
发表回复
HUNTON 2003-09-10
  • 打赏
  • 举报
回复
liyunsong(语法错误) :到ABC三点距离相等的点就是三角形ABC的外心,,你说的圆就是三角形ABC的外接圆啊。我不是都已经给出计算外心公式了么?
fakebear 2003-09-09
  • 打赏
  • 举报
回复
为什么要弧形的呢?就像你说的,小怪物飞行路线如果那么严格的成一个圆弧形,一定不好看。还不如直接从x1直线到x2,然后在直线到x3呢。中间稍微加点随机的方便改变也可以呀。
liyunsong 2003-09-09
  • 打赏
  • 举报
回复
我来告诉你一个最简单的方法:求一个到这三个点距离相等的点,然后以此点为圆心画弧,这个弧就能够同时过这三个点,但是这个圆心怎么求呢?比如:ABC三个顺序排列的不在同一条直线上的点,AB的中垂线与BC的中垂线的交点就是这个圆心.就这样了!
HUNTON 2003-09-08
  • 打赏
  • 举报
回复
最新计算成果:
定义:设平面上的三点A(x1,y1),B(x2,y2),C(x3,y3),定义
|x1 x2 x3|
S(A,B,C) = |y1 y2 y3|
|1 1 1 |

已知三角形的三个顶点为A(x1,y1),B(x2,y2),C(x3,y3),则该三角形的外心为:

S((x1*x1+y1*y1, y1), (x2*x2+y2*y2, y2), (x3*x3+y3*y3, y3))
x0 = -----------------------------------------------------------
2*S(A,B,C)

S((x1,x1*x1+y1*y1), (x2, x2*x2+y2*y2), (x3, x3*x3+y3*y3))
y0 = -----------------------------------------------------------
2*S(A,B,C)
得到外心就可以得到外接圆半径了,这样就可以得到圆弧咯。
zzwu 2003-09-07
  • 打赏
  • 举报
回复
随机产生的3个点P1,P2,P3, 分2种情况
1. 如果P1,P2,P3在一直线上,则无法形成一条弧线,或者说,形成的是一个半径为无穷大的园弧线
2. 如果P1,P2,P3不在一直线上,方法如下:
* 连P1-P2, 求它们的中垂线 m1,
* 连P2-P3, 求它们的中垂线 m2,
* 求m1,m2的交点O,则O就是所求圆弧的圆心,圆弧的半径R=OP1(或OP2,OP3)
* 下面就好办了:以O为圆心,R为半径,OP1为起始角,OP3为终止角,画出圆弧.
flyingakain 2003-09-07
  • 打赏
  • 举报
回复
弄好了给咱粘一份,兄弟,嘿嘿
zhhuang2002 2003-09-07
  • 打赏
  • 举报
回复
Bezier曲线
cywater2000 2003-09-06
  • 打赏
  • 举报
回复
去看<计算机图形学>

里面有很多相关算法
husarb 2003-09-02
  • 打赏
  • 举报
回复
我给一个简单的差值公式:抛物差值公式。这个做出来的就是经过平面上三个点的一条抛物线,如果需要更精确的拟和还有更好的方法。
(x-x1)*(x-x2) (x-x0)*(x-x2) (x-x0)*(x-x1)
公式:L(x)= ---------------*y0 + ---------------*y1 + ---------------*y2
(x0-x1)*(x0-x2) (x1-x0)*(x1-x2) (x2-x0)*(x2-x1)

其中x,y后面带数字的就是该点的坐标,做完后就是这条曲线的方程了。
yanfengqiang 2003-09-01
  • 打赏
  • 举报
回复
怎么插值?

bopengbopeng 2003-09-01
  • 打赏
  • 举报
回复
最好用样条插值。
HUNTON 2003-09-01
  • 打赏
  • 举报
回复
三点确定一个圆,用圆弧拟合
寻开心 2003-08-30
  • 打赏
  • 举报
回复
ax×x + b×x + c = 0
自己带入解方程吧,求出abc参数,三个点确定的一个二次曲线就是这么算的。
crazyyao 2003-08-29
  • 打赏
  • 举报
回复
给定三个点可以确定一条两次曲线,如果再给定开始点的斜率,就可以确定一条三次曲线
klbt 2003-08-29
  • 打赏
  • 举报
回复
顶。
yanfengqiang 2003-08-29
  • 打赏
  • 举报
回复
我只想知道算法就行!,

一个小怪物从上面的一个点x1,经过中间的一个点x2飞到下面的一个点x3,其中x1,x3是确定的,x2是在x1,x3之间产生。

thanks!
nichotilikai 2003-08-29
  • 打赏
  • 举报
回复
怪问题,你想用什么画?GDI?OPENGL?DX?
一、Qt Creator 的安装和hello world 程序的编写(原创) 1.首先到Qt 的官方网站上下载Qt Creator,这里我们下载windows 版的。 下载地址:http://qt.nokia.com/downloads 如下图我们下载:Download Qt SDK for Windows* (178Mb) 下载完成后,直接安装即可,安装过程中按默认设置即可。 2.运行Qt Creator,首先弹出的是欢迎界面,这里可以打开其自带的各种演示 程序。 3.我们用File->New 菜单来新建工程。 4.这里我们选择Qt4 Gui Application。 5.下面输入工程名和要保存到的文件夹路径。我们这里的工程名为helloworld。 6.这时软件自动添加基本的头文件,因为这个程序我们不需要其他的功能,所以 直接击Next。 7.我们将base class 选为QDialog 对话框类。然后击Next。 8.击Finish,完成工程的建立。 9.我们可以看见工程中的所有文件都出现在列表中了。我们可以直接按下下面的 绿色的run 按钮或者按下Ctrl+R 快捷键运行程序。 10.程序运行会出现空白的对话框,如下图。 11.我们双击文件列表的dialog.ui 文件,便出现了下面所示的图形界面编辑界 面。 12.我们在右边的器件栏里找到Label 标签器件 13.按着鼠标左键将其拖到设计窗口上,如下图。 14.我们双击它,并将其内容改为helloworld。 15.我们在右下角的属性栏里将字体大小由9 改为15。 16.我们拖动标签一角的蓝,将全部文字显示出来。 17.再次按下运行按钮,便会出现helloworld。 到这里helloworld 程序便完成了。 Qt Creator 编译的程序,在其工程文件夹下会有一个debug 文件夹,其中有程序的.exe 可执行文件。但Qt Creator 默认是用动态链接的, 就是可执行程序在运行时需要相应的.dll 文件。我们击生成的.exe 文件,首 先可能显示“没有找到mingwm10.dll,因此这个应用程序未能启动。重新安装 应用程序可能会修复此问题。”表示缺少mingwm10.dll 文件。 解决这个问题我们可以将相应的.dll 文件放到系统 中。在Qt Creator 的安装目录的qt 文件下的bin 文件夹下(我安装在了D 盘, 所以路径是D:\Qt\2009.04\qt\bin),可以找到所有的相关.dll 文件。在这里 找到mingwm10.dll 文件,将其复制到C:\WINDOWS\system 文件夹下,即可。下 面再提示缺少什么dll 文件,都像这样解决就可以了。 二、Qt Creator 编写多窗口程序(原创) 实现功能: 程序开始出现一个对话框,按下按钮后便能进入主窗口,如果直 接关闭这个对话框,便不能进入主窗口,整个程序也将退出。当进入主窗口后, 我们按下按钮,会弹出一个对话框,无论如何关闭这个对话框,都会回到主窗口。 实现原理: 程序里我们先建立一个主工程,作为主界面,然后再建立一个对 话框类,将其加入工程中,然后在程序中调用自己新建的对话框类来实现多窗口。 实现过程: 1.首先新建Qt4 Gui Application 工程,工程名为nGui,Base class 选为QWidget。 建立好后工程文件列表如下图。 2.新建对话框类,如下图,在新建中,选择Qt Designer Form Class。 3.选择Dialog without Buttons。 4.类名设为myDlg。 5.击Finish 完成。注意这里已经默认将其加入到了我们刚建的工程中了。 6.如下图,在mydlg.ui 中拖入一个Push Button,将其上的文本改为“进入主 窗口”,在其属性窗口中将其objectName 改为enterBtn,在下面的Signals and slots editor 中进行信号和槽的关联,其中,Sender 设为enterBtn,Signal 设为clicked(),Receive 设为myDlg,Slot 设为accept()。这样就实现了单击 这个按钮使这个对话框关闭并发出Accepted 信号的功能。下面我们将利用这个 信号。 7.修改主函数main.cpp,如下: #include #include "widget.h" #include "mydlg.h" //加入头文件 int main(int argc, char *argv[]) { QApplication a(argc, argv); Widget w; myDlg my1; //建立自己新建的类的对象my1 if(my1.exec()==QDialog::Accepted) //利用Accepted 信号判 断enterBtn 是否被按下 { w.show(); //如果被按下,显示主窗口 return a.exec(); //程序一直执行,直到主窗口 关闭 } else return 0; //如果没被按下,则不会进入主窗口,整个程 序结束运行 } 主函数必须这么写,才能完成所要的功能。 如果主函数写成下面这样: #include #include "widget.h" #include "mydlg.h" int main(int argc, char *argv[]) { QApplication a(argc, argv); myDlg my1; if(my1.exec()==QDialog::Accepted) { Widget w; w.show(); } return a.exec(); } 这样,因为w 是在if 语句里定义的,所以当if 语句执行完后它就无效了。这样 导致的后果就是,按下enterBtn 后,主界面窗口一闪就没了。如果此时对程序 改动了,再次击运行时,就会出现error: collect2: ld returned 1 exit status 的错误。这是因为虽然主窗口没有显示,但它只是隐藏了,程序并没有 结束,而是在后台运行。所以这时改动程序,再运行时便会出错。你可以按下调 试栏上面的红色Stop 停止按钮来停止程序运行。你也可以在windows 任务管理 器的进程中将该进程结束,而后再次运行就没问题了,当然先关闭Qt Creator, 而后再重新打开,这样也能解决问题。 如果把程序改为这样: #include #include "widget.h" #include "mydlg.h" int main(int argc, char *argv[]) { QApplication a(argc, argv); myDlg my1; Widget w; if(my1.exec()==QDialog::Accepted) { w.show(); } return a.exec(); } 这样虽然解决了上面主窗口一闪而过的问题,但是,如果在my1 对话框出现的时 候不enterBtn,而是直接关闭对话框,那么此时整个程序应该结束执行,但 是事实是这样的吗?如果你此时对程序进行了改动,再次按下run 按钮,你会发 现又出现了error: collect2: ld returned 1 exit status 的错误,这说明程 序并没有结束,我们可以打开windows 任务管理器,可以看到我们的程序仍在执 行。 因为return a.exec();一句表示只要主窗口界面不退出,那么程 序就会一直执行。所以只有用第一种方法,将该语句也放到if 语句中,而在else 语句中用else return 0; ,这样如果enterBtn 没有被按下,那么程序就会结 束执行了。 到这里,我们就实现了一个界面结束执行,然后弹出另一个 界面的程序。下面我们在主窗口上加一个按钮,按下该按钮,弹出一个对话框, 但这个对话框关闭,不会使主窗口关闭。 8.如下图,在主窗口加入按钮,显示文本为“弹出一个对话框”,在其上击鼠 标右键,在弹出的菜单中选择go to slot。 9.我们选择单击事件clicked()。 10.我们在弹出的槽函数中添加一句: my2.show(); my2 为我们新建对话框类的另一个对象,但是my2 我们还没有定义,所以 在widget.h 文件中添加相应代码,如下,先加入头文件,再加入my2 的定义语 句,这里我们将其放到private 里,因为一般的函数都放在public 里,而变量 都放在private 里。 #ifndef WIDGET_H #define WIDGET_H #include #include "mydlg.h" //包含头文件 namespace Ui { class Widget; } class Widget : public QWidget { Q_OBJECT public: Widget(QWidget *parent = 0); ~Widget(); private: Ui::Widget *ui; myDlg my2; //对my2 进行定义 private slots: void on_pushButton_clicked(); }; #endif // WIDGET_H 到这里,再运行程序,便能完成我们实验要求的功能了。整个程序里,我们用两 种方法实现了信号和槽函数的关联,第一个按钮我们直接在设计器中实现其关 联;第二个按钮我们自己写了槽函数语句,其实图形的设计与直接写代码效果是 一样的。 这个程序里我们实现了两类窗口打开的方式,一个是自身消失而 后打开另一个窗口,一个是打开另一个窗口而自身不消失。可以看到他们实现的 方法是不同的。 三、Qt Creator 登录对话框(原创) 实现功能: 在弹出对话框中填写用户名和密码,按下登录按钮,如果用户名和密码均正确则 进入主窗口,如果有错则弹出警告对话框。 实现原理: 通过上节的多窗口原理实现由登录对话框进入主窗口,而用户名和密码可以用 if 语句进行判断。 实现过程: 1.先新建Qt4 Gui Application 工程,工程名为mainWidget,选用QWidget 作 为Base class,这样便建立了主窗口。文件列表如下: 2.然后新建一个Qt Designer Form Class 类,类名为loginDlg,选用Dialog without Buttons,将其加入上面的工程中。文件列表如下: 3.在logindlg.ui 中设计下面的界面:行输入框为Line Edit。其中用户名后面 的输入框在属性中设置其object Name 为usrLineEdit,密码后面的输入框为 pwdLineEdit,登录按钮为loginBtn,退出按钮为exitBtn。 4.将exitBtn 的单击后效果设为退出程序,关联如下: 5.右击登录按钮选择go to slot,再选择clicked(),然后进入其单击事件的槽 函数,写入一句 void loginDlg::on_loginBtn_clicked() { accept(); } 6.改写main.cpp: #include #include "widget.h" #include "logindlg.h" int main(int argc, char *argv[]) { QApplication a(argc, argv); Widget w; loginDlg login; if(login.exec()==QDialog::Accepted) { w.show(); return a.exec(); } else return 0; } 7.这时执行程序,可实现按下登录按钮进入主窗口,按下退出按钮退出程序。 8.添加用户名密码判断功能。将登陆按钮的槽函数改为: void loginDlg::on_loginBtn_clicked() { if(m_ui->usrLineEdit->text()==tr("qt")&&m_ui->pwdLineEdit->text()==tr ("123456")) //判断用户名和密码是否正确 accept(); else{ QMessageBox::warning(this,tr("Warning"),tr("user name or password error!"),QMessageBox::Yes); //如果不正确,弹出警告对话框 } } 并在logindlg.cpp 中加入#include 的头文件。如果不加这个头文件, QMessageBox 类不可用。 9.这时再执行程序,输入用户名为qt,密码为123456,按登录按钮便能进入主 窗口了,如果输入错了,就会弹出警告对话框。 如果输入错误,便会弹出警告提示框: 10.在logindlg.cpp 的loginDlg 类构造函数里,添上初始化语句,使密码显示 为小黑。 loginDlg::loginDlg(QWidget *parent) : QDialog(parent), m_ui(new Ui::loginDlg) { m_ui->setupUi(this); m_ui->pwdLineEdit->setEchoMode(QLineEdit::Password); } 效果如下: 11.如果输入如下图中的用户名,在用户名前不小心加上了一些空格,结果程序 按错误的用户名对待了。 我们可以更改if 判断语句,使这样的输入也算正确。 void loginDlg::on_loginBtn_clicked() { if(m_ui->usrLineEdit->text().trimmed()==tr("qt")&&m_ui->pwdLineEdit-> text()==tr("123456")) accept(); else{ QMessageBox::warning(this,tr("Warning"),tr("user name or password error!"),QMessageBox::Yes); } } 加入的这个函数的作用就是移除字符串开头和结尾的空白字符。 12.最后,如果输入错误了,重新回到登录对话框时,我们希望可以使用户名和 密码框清空并且光标自动跳转到用户名输入框,最终的登录按钮的单击事件的槽 函数如下: void loginDlg::on_loginBtn_clicked() { if(m_ui->usrLineEdit->text().trimmed()==tr("qt")&&m_ui->pwdLineEdit-> text()==tr("123456")) //判断用户名和密码是否正确 accept(); else{ QMessageBox::warning(this,tr("Warning"),tr("user name or password error!"),QMessageBox::Yes); //如果不正确,弹出警告对话框 m_ui->usrLineEdit->clear();//清空用户名输入框 m_ui->pwdLineEdit->clear();//清空密码输入框 m_ui->usrLineEdit->setFocus();//将光标转到用户名输入框 } } 四、Qt Creator 添加菜单图标(原创) 在下面的几节,我们讲述Qt 的MainWindow 主窗口部件。这一节只讲述怎样在其 上的菜单栏里添加菜单和图标。 1.新建Qt4 Gui Application 工程,将工程命名为MainWindow,其他选项默认 即可。 生成的窗口界面如下图。其中最上面的为菜单栏。 2.我们在Type Here 那里双击,并输入“文件(&F)”,这样便可将其文件菜单的 快捷键设为Alt+F。(注意括号最好用英文半角输入,这样看着美观) 3.输入完按下Enter 键确认即可,然后在子菜单中加入“新建(&N)”,确定后, 效果如下图。 4.我们在下面的动作编辑窗口可以看到新加的“新建”菜单。 5.双击这一条,可打开它的编辑对话框。我们看到Icon 项,这里可以更改“新 建”菜单的图标。 6.我们击后面的...号,进入资源选择器,但现在这里面是空的。所以下面我 们需要给该工程添加外部资源。 7.添加资源有两种方法。一种是直接添加系统提供的资源文件,然后选择所需图 标。另一种是自己写资源文件。我们主要介绍第一种。新建Qt Resources file, 将它命名为menu。其他默认。 8.添加完后如下图。可以看到添加的文件为menu.qrc。 9.我们最好先在工程文件夹里新建一个文件夹,如images,然后将需要的图标 文件放到其中。 10.在Qt Creator 的menu.qrc 文件中,我们击Add 下拉框,选择Add Prefix。 我们可以将生成的/new/prefix 前缀改为其他名字,如/File。 11.然后再选择Add 下拉框,选择Add Files。再弹出的对话框中,我们到新建 的images 文件夹下,将里面的图标文件全部添加过来。 12.添加完成后,我们在Qt Creator 的File 菜单里选择Save All 选项,保存所 做的更改。 13.这时再打开资源选择器,可以看到我们的图标都在这里了。(注意:如果不显 示,可以按一下上面的Reload 按钮) 14.我们将new.png 作为“新建”菜单的图标,然后击Shortcut,并按下 Crtl+N,便能将Crtl+N 作为“新建”菜单的快捷键。 15.这时打开文件菜单,可以看到“新建”菜单已经有图标了。 运行程序后效果如下。 16.我们在工程文件夹下查看建立的menu.qrc 文件,可以用写字板将它打开。 其具体内容如下。 附:第二种添加资源文件的方法。 1.首先右击工程文件夹,在弹出的菜单中选择Add New,添加新文件。也可以用 File 中的添加新文件。 2.我们选择文本文件。 3.将文件名设置为menu.qrc。 4.添加好文件后将其内容修改如下。可以看到就是用第一种方法生成的 menu.qrc 文件的内容。 5.保存文件后,在资源管理器中可以看到添加的图标文件。 五、Qt Creator 布局管理器的使用(原创) 上篇讲解了如何在Qt Creator 中添加资源文件,并且为菜单添加了图标。这次 我们先对那个界面进行一些完善,然后讲解一些布局管理器的知识。 首先对菜单进行完善。 1.我们在上一次的基础上再加入一些常用菜单。 “文件”的子菜单如下图。中间的割线可以击Add Separator 添加。 “编辑”子菜单的内容如下。 “帮助”子菜单的内容如下。 2.我们在动作编辑器中对各个菜单的属性进行设置。 如下图。 3.我们拖动“新建”菜单的图标,将其放到工具栏里。 拖动“新建”菜单的图标。 将其放到菜单栏下面的工具栏里。 4.我们再添加其他几个图标。使用Append Separator 可以添加割线。 5.最终效果如下。如果需要删除图标,可以在图标上击右键选择Remove action 即可。 下面简述一下布局管理器。 (这里主要以垂直布局管理器进行讲解,其他类型管理器用法与之相同,其效 果可自己验证。) 1.在左边的器件栏里拖入三个PushButton 和一个Vertical Layout(垂直布局 管理器)到中心面板。如下图。 2.将这三个按钮放入垂直布局管理器,效果如下。可以看到按钮垂直方向排列, 并且宽度可以改变,但高度没有改变。 3.我们将布局管理器整体选中,按下上面工具栏的Break Layout 按钮,便可取 消布局管理器。(我们当然也可以先将按钮移出,再按下Delete 键将布局管理 器删除。) 4.下面我们改用裂器部件(QSplitter)。 先将三个按钮同时选中,再按下上面工具栏的Lay Out Vertically in Splitter (垂直裂器)。 效果如下图。可以看到按钮的大小可以随之改动。这也就是裂器和布局管理器 的别。 5.其实布局管理器不但能控制器件的布局,还有个很重要的用途是,它能使器件 的大小随着窗口大小的改变而改变。 我们先在主窗口的中心拖入一个文本编辑器Text Edit。 这时直接运行程序,效果如下。可以看到它的大小和位置不会随着窗口改变。 下面我们选中主窗口部件,然后在空白处击鼠标右键,选择Layout->Lay Out in a Grid,使整个主窗口的中心区处于网格布局管理器中。 可以看到,这时文本编辑器已经占据了整个主窗口的中心区。 运行一下程序,可以看到无论怎样拉伸窗口,文本编辑框的大小都会随之改变。 我们在这里一共讲述了三种使用布局管理器的方法,一种是去器件栏添加,一 种是用工具栏的快捷图标,还有一种是使用鼠标右键的选项。 程序中用到的图标是我从Ubuntu 中复制的,可以到 http://www.qtcn.org/bbs/read.php?tid=23252&page=1&toread=1 下载到。 六、Qt Creator 实现文本编辑(原创) 前面已经将界面做好了,这里我们为其添加代码,实现文本编辑的功能。 首先实现新建文件,文件保存,和文件另存为的功能。 (我们先将上次的工程文件夹进行备份,然后再对其进行修改。在写较大的程序 时,经常对源文件进行备份,是个很好的习惯。) 在开始正式写程序之前,我们先要考虑一下整个流程。因为我们要写记事本一 样的软件,所以最好先打开windows 中的记事本,进行一些简单的操作,然后 考虑怎样去实现这些功能。再者,再强大的软件,它的功能也是一个一个加上 去的,不要设想一下子写出所有的功能。我们这里先实现新建文件,保存文件, 和文件另存为三个功能,是因为它们联系很紧,而且这三个功能总的代码量也 不是很大。 因为三个功能之间的关系并不复杂,所以我们这里便不再画流程图,而只是简 单描述一下。 新建文件,那么如果有正在编辑的文件,是否需要保存呢? 如果需要进行保存,那这个文件以前保存过吗?如果没有保存过,就应该先将其 另存为。 下面开始按这些关系写程序。 1.打开Qt Creator,在File 菜单中选择Open,然后在工程文件夹中打开 MainWindow.pro 工程文件。 先在main.cpp 文件中加入以下语句,让程序中可以使用中文。 在其中加入#include 头文件包含,再在主函数中加入下面一行: QTextCodec::setCodecForTr(QTextCodec::codecForLocale()); 这样在程序中使用中文,便能在运行时显示出来了。更改后文件如下图。 2.在mainwindow.h 文件中的private 下加入以下语句。 bool isSaved; //为true 时标志文件已经保存,为false 时标志文件尚未保存 QString curFile; //保存当前文件的文件名 void do_file_New(); //新建文件 void do_file_SaveOrNot(); //修改过的文件是否保存 void do_file_Save(); //保存文件 void do_file_SaveAs(); //文件另存为 bool saveFile(const QString& fileName); //存储文件 这些是变量和函数的声明。其中isSaved 变量起到标志的作用,用它来标志文件 是否被保存过。然后我们再在相应的源文件里进行这些函数的定义。 3.在mainwindow.cpp 中先加入头文件#include ,然后在构造函数里添 加以下几行代码。 isSaved = false; //初始化文件为未保存过状态 curFile = tr("未命名.txt"); //初始化文件名为“未命名.txt” setWindowTitle(curFile); //初始化主窗口的标题 这是对主窗口进行初始化。效果如下。 4.然后添加“新建”操作的函数定义。 void MainWindow::do_file_New() //实现新建文件的功能 { do_file_SaveOrNot(); isSaved = false; curFile = tr("未命名.txt"); setWindowTitle(curFile); ui->textEdit->clear(); //清空文本编辑器 ui->textEdit->setVisible(true); //文本编辑器可见 } 新建文件,先要判断正在编辑的文件是否需要保存。然后将新建的文件标志为未 保存过状态。 5.再添加do_file_SaveOrNot 函数的定义。 void MainWindow::do_file_SaveOrNot() //弹出是否保存文件对话框 { if(ui->textEdit->document()->isModified()) //如果文件被更改过,弹出保 存对话框 { QMessageBox box; box.setWindowTitle(tr("警告")); box.setIcon(QMessageBox::Warning); box.setText(curFile + tr(" 尚未保存,是否保存?")); box.setStandardButtons(QMessageBox::Yes | QMessageBox::No); if(box.exec() == QMessageBox::Yes) //如果选择保存文件,则执行保存操作 do_file_Save(); } } 这个函数实现弹出一个对话框,询问是否保存正在编辑的文件。 6.再添加“保存”操作的函数定义。 void MainWindow::do_file_Save() //保存文件 { if(isSaved){ //如果文件已经被保存过,直接保存文件 saveFile(curFile); } else{ do_file_SaveAs(); //如果文件是第一次保存,那么调用另存为 } } 对文件进行保存时,先判断其是否已经被保存过,如果没有被保存过,就要先对 其进行另存为操作。 7.下面是“另存为”操作的函数定义。 void MainWindow::do_file_SaveAs() //文件另存为 { QString fileName = QFileDialog::getSaveFileName(this,tr("另存为 "),curFile); //获得文件名 if(!fileName.isEmpty()) //如果文件名不为空,则保存文件内容 { saveFile(fileName); } } 这里弹出一个文件对话框,显示文件另存为的路径。 8.下面是实际文件存储操作的函数定义。 bool MainWindow::saveFile(const QString& fileName) //保存文件内容,因为可能保存失败,所以具有返回值,来表明是否保存成功 { QFile file(fileName); if(!file.open(QFile::WriteOnly | QFile::Text)) //以只写方式打开文件,如果打开失败则弹出提示框并返回 { QMessageBox::warning(this,tr("保存文件"), tr("无法保存文件 %1:\n %2").arg(fileName) .arg(file.errorString())); return false; } //%1,%2 表示后面的两个arg 参数的值 QTextStream out(&file); //新建流对象,指向选定的文件 out << ui->textEdit->toPlainText(); //将文本编辑器里的内容以纯文本 的形式输出到流对象中 isSaved = true; curFile = QFileInfo(fileName).canonicalFilePath(); //获得文件的标准路 径 setWindowTitle(curFile); //将窗口名称改为现在窗口的路径 return true; } 这个函数实现将文本文件进行存储。下面我们对其中的一些代码进行讲解。 QFile file(fileName);一句,定义了一个QFile 类的对象file,其中filename 表明这个文件就是我们保存的的文件。然后我们就可以用file 代替这个文件, 来进行一些操作。Qt 中文件的操作和C,C++很相似。对于QFile 类对象怎么使 用,我们可以查看帮助。 击Qt Creator 最左侧的Help,在其中输入QFile, 在搜索到的列表中选择QFile 即可。这时在右侧会显示出QFile 类中所有相关信 息以及他们的用法和说明。 // 我们往下拉,会发现下面有关于怎么读取文件的示例代码。 // // 再往下便能看到用QTextStream 类对象,进行字符串输入的例子。下面也提到了 QFileInfo 和QDir 等相关的类,我们可以击它们去看一下具体的使用说明。 // 上面只是做了一个简单的说明。以后我们对自己不明白的类都可以去帮助里进行 查找,这也许是我们以后要做的最多的一件事了。对于其中的英文解释,我们最 好想办法弄明白它的大意,其实网上也有一些中文的翻译,但最好还是从一开始 就尝试着看英文原版的帮助,这样以后才不会对中文翻译产生依赖。 我们这次只是很简单的说明了一下怎样使用帮助文件,这不表明 它不重要,而是因为这里不可能将每个类的帮助都解释一遍,没有那么多时间, 也没有那么大的篇幅。而更重要的是因为,我们这个教程只是引你入门,所以很 多东西需要自己去尝试。 在以后的教程里,如果不是特殊情况,就不会再对其中的类进行 详细解释,文章中的重是对整个程序的描述,其中不明白的类,自己查看帮助。 9.双击mainwindow.ui 文件,在图形界面窗口下面的Action Editor 动作编辑 器里,我们右击“新建”菜单一条,选择Go to slot,然后选择triggered(), 进入其触发事件槽函数。 同理,进入其他两个菜单的槽函数,将相应的操作的函数写入槽函数中。如下。 void MainWindow::on_action_New_triggered() //信号和槽的关联 { do_file_New(); } void MainWindow::on_action_Save_triggered() { do_file_Save(); } void MainWindow::on_action_SaveAs_triggered() { do_file_SaveAs(); } 这时击运行,就能够实现新建文件,保存文件,文件另存为的功能了。 然后实现打开,关闭,退出,撤销,复制,剪切,粘贴的功能。 先备份上次的工程文件,然后再将其打开。 1.先在mainwindow.h 文件中加入函数的声明。 void do_file_Open(); //打开文件 bool do_file_Load(const QString& fileName); //读取文件 2.再在mainwindow.cpp 文件中写函数的功能实现。 void MainWindow::do_file_Open()//打开文件 { do_file_SaveOrNot();//是否需要保存现有文件 QString fileName = QFileDialog::getOpenFileName(this); //获得要打开的文件的名字 if(!fileName.isEmpty())//如果文件名不为空 { do_file_Load(fileName); } ui->textEdit->setVisible(true);//文本编辑器可见 } bool MainWindow::do_file_Load(const QString& fileName) //读取文件 { QFile file(fileName); if(!file.open(QFile::ReadOnly | QFile::Text)) { QMessageBox::warning(this,tr("读取文件"),tr("无法读取文件 %1:\n%2.").arg(fileName).arg(file.errorString())); return false; //如果打开文件失败,弹出对话框,并返回 } QTextStream in(&file); ui->textEdit->setText(in.readAll()); //将文件中的所有内容都 写到文本编辑器中 curFile = QFileInfo(fileName).canonicalFilePath(); setWindowTitle(curFile); return true; } 上面的打开文件函数与文件另存为函数相似,读取文件的函数与文件存储函数相 似。 3.然后按顺序加入更菜单的关联函数,如下。 void MainWindow::on_action_Open_triggered() //打开操作 { do_file_Open(); } // void MainWindow::on_action_Close_triggered() //关闭操作 { do_file_SaveOrNot(); ui->textEdit->setVisible(false); } // void MainWindow::on_action_Quit_triggered() //退出操作 { on_action_Close_triggered(); //先执行关闭操作 qApp->quit(); //再退出系统,qApp 是指向应用程序的全局指针 } // void MainWindow::on_action_Undo_triggered() //撤销操作 { ui->textEdit->undo(); } // void MainWindow::on_action_Cut_triggered() //剪切操作 { ui->textEdit->cut(); } // void MainWindow::on_action_Copy_triggered() //复制操作 { ui->textEdit->copy(); } // void MainWindow::on_action_Past_triggered() //粘贴操作 { ui->textEdit->paste(); } 因为复制,撤销,全选,粘贴,剪切等功能,是TextEdit 默认就有的,所以我 们只需调用一下相应函数就行。 到这里,除了查找和帮助两个菜单的功能没有加上以外,其他功能都已经实现了。 七、Qt Creator 实现文本查找(原创) 现在加上查找菜单的功能。因为这里要涉及关于Qt Creator 的很多实用功能, 所以单独用一篇文章来介绍。 以前都用设计器设计界面,而这次我们用代码实现一个简单的查找对话框。对于 怎么实现查找功能的,我们详细地步说明了怎么进行类中方法的查找和使用。 其中也将Qt Creator 智能化的代码补全功能和程序中函数的声明位置和定义位 置间的快速切换进行了介绍。 1.首先还是保存以前的工程,然后再将其打开。 我们发现Qt Creator 默认的字体有小,可以按下Ctrl 键的同时按两下+键, 来放大字体。也可以选择Edit->Advanced->Increase Font Size。 2.在mainwindow.h 中加入#include 的头文件包含,在private 中 添加 QLineEdit *find_textLineEdit; //声明一个行编辑器,用于输入要查找的内容 在private slots 中添加 void show_findText(); 在该函数中实现查找字符串的功能。 3.我们进入查找菜单的触发事件槽函数,更改如下。 void MainWindow::on_action_Find_triggered() { QDialog *findDlg = new QDialog(this); //新建一个对话框,用于查找操作,this 表明它的父窗口是MainWindow。 findDlg->setWindowTitle(tr("查找")); //设置对话框的标题 find_textLineEdit = new QLineEdit(findDlg); //将行编辑器加入到新建的查找对话框中 QPushButton *find_Btn = new QPushButton(tr("查找下一个"),findDlg); //加入一个“查找下一个”的按钮 QVBoxLayout* layout = new QVBoxLayout(findDlg); layout->addWidget(find_textLineEdit); layout->addWidget(find_Btn); //新建一个垂直布局管理器,并将行编辑器和按钮加入其中 findDlg ->show(); //显示对话框 connect(find_Btn,SIGNAL(clicked()),this,SLOT(show_findText())); //设置“查找下一个”按钮的单击事件和其槽函数的关联 } 这里我们直接用代码生成了一个对话框,其中一个行编辑器可以输入要查找的字 符,一个按钮可以进行查找操作。我们将这两个部件放到了一个垂直布局管理器 中。然后显示这个对话框。并设置了那个按钮单击事件与show_findText()函数 的关联。 5.下面我们开始写实现查找功能的show_findText()函数。 void MainWindow::show_findText()//“查找下一个”按钮的槽函数 { QString findText = find_textLineEdit->text(); //获取行编辑器中的内容 } 先用一个QString 类的对象获得要查找的字符。然后我们一步一步写查找操作的 语句。 6.在下一行写下ui,然后直接按下键盘上的“<.”键,这时系统会根据是否是 指针对象而自动生成“->”或“.”,因为ui 是指针对象,所以自动生成“->” 号,而且弹出了ui 中的所有部件名称的列表。如下图。 7.我们用向下的方向键选中列表中的textEdit。或者我们可以先输入text,这 时能缩减列表的内容。 8.如上图我们将鼠标放到textEdit 上,这时便出现了textEdit 的类名信息, 且后面出现一个F1 按键。我们按下键盘上的F1,便能出现textEdit 的帮助。 9.我们在帮助中向下拉,会发现这里有一个find 函数。 10.我们击find,查看其详细说明。 11.可以看到find 函数可以实现文本编辑器中字符串的查找。其中有一个 FindFlags 的参数,我们击它查看其说明。 12.可以看到它是一个枚举变量(enum),有三个选项,第一项是向后查找(即 查找光标以前的内容,这里的前后是相对的说法,比如第一行已经用完了,光 标在第二行时,把第一行叫做向后。),第二项是区大小写查找,第三项是 查找全部。 13.我们选用第一项,然后写出下面的语句。 ui->textEdit->find(findText,QTextDocument::FindBackward); //将行编辑器中的内容在文本编辑器中进行查找 当我们刚打出“f”时,就能自动弹出textEdit 类的相关属性和方法。 可以看到,当写完函数名和第一个“(”后,系统会自动显示出该函数的函数原 型,这样可以使我们减少出错。 14.这时已经能实现查找的功能了。但是我们刚才看到find 的返回值类型是bool 型,而且,我们也应该为查找不到字符串作出提示。 if(!ui->textEdit->find(findText,QTextDocument::FindBackward)) { QMessageBox::warning(this,tr("查找"),tr("找不到 %1") .arg(findText); } 因为查找失败返回值是false,所以if 条件加了“!”号。在找不到时弹出警 告对话框。 15.到这里,查找功能就基本上写完了。show_findText()函数的内容如下。 我们会发现随着程序功能的增强,其中的函数也会越来越多,我们都会为查找 某个函数的定义位置感到头疼。而在Qt Creator 中有几种快速定位函数的方法, 我们这里讲解三种。 第一,在函数声明的地方直接跳转到函数定义的地方。 如在do_file_Load 上击鼠标右键,在弹出的菜单中选择Follow Symbol under Cursor 或者下面的Switch between Method Declaration/Definition。 这时系统就会自动跳转到函数定义的位置。如下图。 第二,快速查找一个文件里的所有函数。 我们可以击窗口最上面的下拉框,这里会显示本文件中所有函数的列表。 第三,利用查找功能。 1.我们先将鼠标定位到一个函数名上。 2.然后选择Edit->Find/Replace->Find Dialog。 3.这时会出现一个查找对话框,可以看到要查找的函数名已经写在里面了。 4.当我们按下Search 按钮后,会在查找结果窗口显示查找到的结果。 5.我们击第二个文件。会发现在这个文件中有两处关键字是高亮显示。 6.我们双击第二项,就会自动跳转到函数的定义处。 文章讲到这里,我们已经很详细地说明了怎样去使用一个类里面没有用过的方法 函数;也说明了Qt Creator 中的一些便捷操作。可以看到,Qt Creator 开发环 境,有很多很人性化的设计,我们应该熟练应用它们。 在以后的文章中,我们不会再很详细地去用帮助来说明一个函数是 怎么来的,该怎么用,这些应该自己试着去查找。 八、Qt Creator 实现状态栏显示(原创) 在程序主窗口Mainwindow 中,有菜单栏,工具栏,中心部件和状态栏。前面几 个已经讲过了,这次讲解状态栏的使用。 程序中有哪些不明白的类或函数,请自己查看帮助。 1.我们在mainwindow.h 中做一下更改。 加入头文件包含: #include 加入私有变量和函数: QLabel* first_statusLabel; //声明两个标签对象,用于显示状态信息 QLabel* second_statusLabel; void init_statusBar(); //初始化状态栏 加入一个槽函数声明:void do_cursorChanged(); //获取光标位置信息 2.在mainwindow.cpp 中加入状态栏初始化函数的定义。 void MainWindow::init_statusBar() { QStatusBar* bar = ui->statusBar; //获取状态栏 first_statusLabel = new QLabel; //新建标签 first_statusLabel->setMinimumSize(150,20); //设置标签最小尺寸 first_statusLabel->setFrameShape(QFrame::WinPanel); //设置标签形状 first_statusLabel->setFrameShadow(QFrame::Sunken); //设置标签阴影 second_statusLabel = new QLabel; second_statusLabel->setMinimumSize(150,20); second_statusLabel->setFrameShape(QFrame::WinPanel); second_statusLabel->setFrameShadow(QFrame::Sunken); bar->addWidget(first_statusLabel); bar->addWidget(second_statusLabel); first_statusLabel->setText(tr("欢迎使用文本编辑器")); //初始化内容 second_statusLabel->setText(tr("yafeilinux 制作!")); } 这里将两个标签对象加入到了主窗口的状态栏里,并设置了他们的外观和初值。 3.在构造函数里调用状态栏初始化函数。 init_statusBar(); 这时运行程序,效果如下。 4.在mainwindow.cpp 中加入获取光标位置的函数的定义。 void MainWindow::do_cursorChanged() { int rowNum = ui->textEdit->document()->blockCount(); //获取光标所在行的行号 const QTextCursor cursor = ui->textEdit->textCursor(); int colNum = cursor.columnNumber(); //获取光标所在列的列号 first_statusLabel->setText(tr("%1 行 %2 列").arg(rowNum).arg(colNum)); //在状态栏显示光标位置 } 这个函数可获取文本编辑框中光标的位置,并显示在状态栏中。 5.在构造函数添加光标位置改变信号的关联。 connect(ui->textEdit,SIGNAL(cursorPositionChanged()),this,SLOT(do_cur sorChanged())); 这时运行程序。效果如下。 6.在do_file_Load 函数的最后添加下面语句。 second_statusLabel->setText(tr("打开文件成功")); 7.在saveFile 函数的最后添加以下语句。 second_statusLabel->setText(tr("保存文件成功")); 8.在on_action_Find_triggered 函数的后面添加如下语句。 second_statusLabel->setText(tr("正在进行查找")); 9.在on_action_Close_triggered 函数最后添加如下语句。 first_statusLabel->setText(tr("文本编辑器已关闭")); second_statusLabel->setText(tr("yafeilinux 制作!")); 到这里整个文本编辑器的程序就算写完了。我们这里没有写帮助菜单的功能实 现,大家可以自己添加。而且程序中也有很多漏洞和不完善的地方,如果有兴 趣,大家也可以自己修改。因为时间和篇幅的原因,我们这里就不再过多的讲 述。 九、Qt Creator 中鼠标键盘事件的处理实现自定义鼠标指针(原创) 我们前面一直在说信号,比方说用鼠标按了一下按钮,这样就会产生一个按钮的 单击信号,然后我们可以在相应的槽函数里进行相应功能的设置。其实在按下鼠 标后,程序要先接收到鼠标按下的事件,然后将这个事件按默认的设置传给按钮。 可以看出,事件和信号并不是一回事,事件比信号更底层。而我们以前把单击按 钮也叫做事件,这是不确切的,不过大家都知道是什么意思,所以当时也没有细 。 Qt 中的事件可以在QEvent 中查看。下面我们只是找两个例子来进行简单的演示。 1.还是先建立一个Qt4 Gui Application 工程,我这里起名为event。 2.添加代码,让程序中可以使用中文。 即在main.cpp 文件中加入#include 的头文件包含。 再在下面的主函数里添加 QTextCodec::setCodecForTr(QTextCodec::codecForLocale()); 3.在mainwindow.h 文件中做一下更改。 添加#include 头文件。因为这样就包含了QtGui 中所有的子文件。 在public 中添加两个函数的声明 void mouseMoveEvent(QMouseEvent *); void keyPressEvent(QKeyEvent *); 4.我们在mainwindow.ui 中添加一个Label 和一个PushButton,将他们拉长, 因为一会要在上面显示标语。 5.在mainwindow.cpp 中的构造函数里添加两个部件的显示文本。 ui->label->setText(tr("按下键盘上的A 键试试!")); ui->pushButton->setText(tr("按下鼠标的一个键,然后移动鼠标试试")); 6.然后在下面进行两个函数的定义。 /*以下是鼠标移动事件*/ void MainWindow::mouseMoveEvent(QMouseEvent *m) {//这里的函数名和参数不能更改 QCursor my(QPixmap("E:/Qt/Qt-Creator-Example/event/time.png")); //为鼠标指针选择图片,注意这里要用绝对路径,且要用“/”,而不能用“\” QApplication::setOverrideCursor(my); //将鼠标指针更改为自己设置的图片 int x = m->pos().x(); int y = m->pos().y(); //获取鼠标现在的位置坐标 ui->pushButton->setText(tr("鼠标现在的坐标是(%1,%2), 哈哈好玩吧 ").arg(x).arg(y)); //将鼠标的位置坐标显示在按钮上 ui->pushButton->move(m->pos()); //让按钮跟随鼠标移动 } /*以下是键盘按下事件*/ void MainWindow::keyPressEvent(QKeyEvent *k) { if(k->key() == Qt::Key_A) //判断是否是A 键按下 { ui->label->setPixmap(QPixmap("E:/Qt/Qt-Creator-Example/event/linux.jp g")); ui->label->resize(100,100); //更改标签图片和大小 } } 注意:这两个函数不是自己新建的,而是对已有函数的重定义,所有函数名和参 数都不能改。第一个函数对鼠标移动事件进行了重写。其中实现了鼠标指针的更 改,和按钮跟随鼠标移动的功能。 第二个函数对键盘的A 键按下实现了新的功能。 效果如下。 按下鼠标的一个键,并移动鼠标。 按下键盘上的A 键。 十、Qt Creator 中实现定时器和产生随机数(原创) 有两种方法实现定时器。 第一种。自己建立关联。 1.新建Gui 工程,工程名可以设置为timer。并在主界面上添加一个标签label, 并设置其显示内容为“0000-00-00 00:00:00 星期日”。 2.在mainwindow.h 中添加槽函数声明。 private slots: void timerUpDate(); 3.在mainwindow.cpp 中添加代码。 添加#include 的头文件包含,这样就包含了QtCore 下的所有文件。 构造函数里添加代码: QTimer *timer = new QTimer(this); //新建定时器 connect(timer,SIGNAL(timeout()),this,SLOT(timerUpDate())); //关联定时器计满信号和相应的槽函数 timer->start(1000); //定时器开始计时,其中1000 表示1000ms 即1 秒 4.然后实现更新函数。 void MainWindow::timerUpDate() { QDateTime time = QDateTime::currentDateTime(); //获取系统现在的时间 QString str = time.toString("yyyy-MM-dd hh:mm:ss dddd"); //设置系统时间显示格式 ui->label->setText(str); //在标签上显示时间 } 5.运行程序,效果如下。 第二种。使用事件。(有像单片机中的定时器啊) 1.新建工程。在窗口上添加两个标签。 2.在main.cpp 中添加代码,实现中文显示。 #include QTextCodec::setCodecForTr(QTextCodec::codecForLocale()); 3.在mainwindow.h 中添加代码。 void timerEvent(QTimerEvent *); 4.在mainwindow.cpp 中添加代码。 添加头文件#include 在构造函数里添加以下代码。 startTimer(1000); //其返回值为1,即其timerId 为1 startTimer(5000);//其返回值为2,即其timerId 为2 startTimer(10000); //其返回值为3,即其timerId 为3 添加了三个定时器,它们的timerId 别为1,2,3。注意,第几个定时器的返 回值就为几。所以要注意定时器顺序。 在下面添加函数实现。 void MainWindow::timerEvent(QTimerEvent *t) //定时器事件 { switch(t->timerId()) //判断定时器的句柄 { case 1 : ui->label->setText(tr("每秒产生一个随机数: %1").arg(qrand()%10));break; case 2 : ui->label_2->setText(tr("5 秒后软件将关闭"));break; case 3 : qApp->quit();break; //退出系统 } } 这里添加了三个定时器,并都在定时器事件中判断它们,然后执行相应的功能。 这样就不用每个定时器都写一个关联函数和槽函数了。 随机数的实现: 上面程序中的qrand(),可以产生随机数,qrand()%10 可以产生0-9 之间的随机 数。要想产生100 以内的随机数就%100。以此类推。 但这样每次启动程序后,都按同一种顺序产生随机数。为了实现每次启动程序产 生不同的初始值。我们可以使用qsrand(time(0));实现设置随机数的初值,而 程序每次启动时time(0)返回的值都不同,这样就实现了产生不同初始值的功 能。 我们将qsrand(time(0));一句加入构造函数里。 程序最终运行效果如下。 十一、Qt 2D 绘图(一)绘制简单图形(原创) 声明:本文原创于yafeilinux 的百度博客,http://hi.baidu.com/yafeilinux 转载请注明出处。 说明:以后使用的环境为基于Qt 4.6 的Qt Creator 1.3.0 windows 版本 本文介绍在窗口上绘制最简单的图形的方法。 1.新建Qt4 Gui Application 工程,我这里使用的工程名为painter01,选用 QDialog 作为Base class 2.在dialog.h 文件中声明重绘事件函数void paintEvent(QPaintEvent *); 3.在dialog.cpp 中添加绘图类QPainter 的头文件包含#include 4.在下面进行该函数的重定义。 void Dialog::paintEvent(QPaintEvent *) { QPainter painter(this); painter.drawLine(0,0,100,100); } 其中创建了QPainter 类对象,它是用来进行绘制图形的,我们这里画了一条线 Line,其中的参数为线的起(0,0),和终100100)。这里的数值指的 是像素,详细的坐标设置我们以后再讲,这里知道(0,0)指的是窗口的左上 角即可。运行效果如下: 5.在qt 的帮助里可以查看所有的绘制函数,而且下面还给出了相关的例子。 6.我们下面将几个知识说明一下,帮助大家更快入门。 将函数改为如下: void Dialog::paintEvent(QPaintEvent *) { QPainter painter(this); QPen pen; //画笔 pen.setColor(QColor(255,0,0)); QBrush brush(QColor(0,255,0,125)); //画刷 painter.setPen(pen); //添加画笔 painter.setBrush(brush); //添加画刷 painter.drawRect(100,100,200,200); //绘制矩形 } 这里的pen 用来绘制边框,brush 用来进行封闭区域的填充,QColor 类用来提供 颜色,我们这里使用了rgb 方法来生成颜色,即(red,green,blue),它们取 值别是0-255,例如(255,0,0)表示红色,而全0 表示黑色,全255 表示 白色。后面的(0,255,0,125),其中的125 是透明度(alpha)设置,其值 也是从0 到255,0 表示全透明。最后将画笔和画刷添加到painter 绘制设备中, 画出图形。这里的Rect 是长方形,其中的参数为(100100)表示起始坐标, 200,200 表示长和宽。效果如下: 7.其实画笔和画刷也有很多设置,大家可以查看帮助。 QPainter painter(this); QPen pen(Qt::DotLine); QBrush brush(Qt::blue); brush.setStyle(Qt::HorPattern); painter.setPen(pen); painter.setBrush(brush); painter.drawRect(100,100,200,200); 这里我们设置了画笔的风格为线,画刷的风格为并行横线,效果如下: 在帮助里可以看到所有的风格。 我们这里用了Qt::blue,Qt 自定义的几个颜色如下: 8.画弧线,这是帮助里的一个例子。 QRectF rectangle(10.0, 20.0, 80.0, 60.0); //矩形 int startAngle = 30 * 16; //起始角度 int spanAngle = 120 * 16; //跨越度数 QPainter painter(this); painter.drawArc(rectangle, startAngle, spanAngle); 这里要说明的是,画弧线时,角度被成了十六之一,就是说,要想为30 度, 就得是30*16。它有起始角度和跨度,还有位置矩形,要想画出自己想要的弧线, 就要有一定的几何知识了。这里就不再祥述。 十二、Qt 2D 绘图(二)渐变填充(原创) 声明:本文原创于yafeilinux 的百度博客,http://hi.baidu.com/yafeilinux 转载请注明出处。 在qt 中提供了三种渐变方式,别是线性渐变,圆形渐变和圆锥渐变。如果能 熟练应用它们,就能设计出炫目的填充效果。 线性渐变: 1.更改函数如下: void Dialog::paintEvent(QPaintEvent *) { QPainter painter(this); QLinearGradient linearGradient(100,150,300,150); //从100,150)开始到(300,150)结束,确定一条直线 linearGradient.setColorAt(0,Qt::red); linearGradient.setColorAt(0.2,Qt::black); linearGradient.setColorAt(0.4,Qt::yellow); linearGradient.setColorAt(0.6,Qt::white); linearGradient.setColorAt(0.8,Qt::green); linearGradient.setColorAt(1,Qt::blue); //将直线开始设为0,终设为1,然后段设置颜色 painter.setBrush(linearGradient); painter.drawRect(100,100,200,100); //绘制矩形,线性渐变线正好在矩形的水平中心线上 } 效果如下: 圆形渐变: 1.更改函数内容如下: QRadialGradient radialGradient(200,100,100,200,100); //其中参数别为圆形渐变的圆心(200,100),半径100,和焦(200, 100) //这里让焦和圆心重合,从而形成从圆心向外渐变的效果 radialGradient.setColorAt(0,Qt::black); radialGradient.setColorAt(1,Qt::yellow); //渐变从焦向整个圆进行,焦为起始0,圆的边界为1 QPainter painter(this); painter.setBrush(radialGradient); painter.drawEllipse(100,0,200,200); //绘制圆,让它正好和上面的圆形渐变的圆重合 效果如下: 2.要想改变填充的效果,只需要改变焦的位置和渐变的颜色位置即可。 改变焦位置:QRadialGradient radialGradient(200,100,100,100,100); 效果如下: 锥形渐变: 1.更改函数内容如下: //圆锥渐变 QConicalGradient conicalGradient(50,50,0); //圆心为(50,50),开始角度为0 conicalGradient.setColorAt(0,Qt::green); conicalGradient.setColorAt(1,Qt::white); //从圆心的0 度角开始逆时针填充 QPainter painter(this); painter.setBrush(conicalGradient); painter.drawEllipse(0,0,100,100); 效果如下: 2.可以更改开始角度,来改变填充效果 QConicalGradient conicalGradient(50,50,30); 开始角度设置为30 度,效果如下: 其实三种渐变的设置都在于焦和渐变颜色的位置,如果想设计出漂亮的渐变 效果,还要有美术功底啊! 十二、Qt 2D 绘图(三)绘制文字(原创) 声明:本文原创于yafeilinux 的百度博客,http://hi.baidu.com/yafeilinux 转载请注明出处。 接着上一次的教程,这次我们学习在窗体上绘制文字。 1.绘制最简单的文字。 我们更改重绘函数如下: void Dialog::paintEvent(QPaintEvent *) { QPainter painter(this); painter.drawText(100,100,"yafeilinux"); } 我们在(100100)的位置显示了一行文字,效果如下。 2.为了更好的控制字体的位置。我们使用另一个构造函数。在帮助里查看 drawText,如下。 这里我们看到了构造函数的原型和例子。其中的flags 参数可以控制字体在矩形 中的位置。我们更改函数内容如下。 void Dialog::paintEvent(QPaintEvent *) { QPainter painter(this); QRectF ff(100,100,300,200); //设置一个矩形 painter.drawRect(ff); //为了更直观地看到字体的位置,我们绘制出这个矩形 painter.setPen(QColor(Qt::red)); //设置画笔颜色为红色 painter.drawText(ff,Qt::AlignHCenter,"yafeilinux"); //我们这里先让字体水平居中 } 效果如下。 可以看到字符串是在最上面水平居中的。如果想让其在矩形正中间,我们可以使 用Qt::AlignCenter。 这里我们也可以使用两个枚举变量进行按位与操作,例如可以使用 Qt::AlignBottom|Qt::AlignHCenter 实现让文字显示在矩形下面的正中间。效 果如下。 对于较长的字符串,我们也可以利用“\n”进行换行,例如"yafei\nlinux"。效 果如下。 3.如果要使文字更美观,我们就需要使用QFont 类来改变字体。先在帮助中查 看一下这个类。 可以看到它有好几个枚举变量来设置字体。下面的例子我们对主要的几个选项进 行演示。 更改函数如下。 void Dialog::paintEvent(QPaintEvent *) { QFont font("Arial",20,QFont::Bold,true); //设置字体的类型,大小,加粗,斜体 font.setUnderline(true); //设置下划线 font.setOverline(true); //设置上划线 font.setCapitalization(QFont::SmallCaps); //设置大小写 font.setLetterSpacing(QFont::AbsoluteSpacing,5); //设置间距 QPainter painter(this); painter.setFont(font); //添加字体 QRectF ff(100,100,300,200); painter.drawRect(ff); painter.setPen(QColor(Qt::red)); painter.drawText(ff,Qt::AlignCenter,"yafeilinux"); } 效果如下。 这里的所有字体我们可以在设计器中进行查看。如下。 基于Qt 4.6 的Qt Creator 1.3.0 环境变量设置(原创) 如果你以前安装过visual studio 2005 之类的软件,那么装上Qt Creator 1.3.0 后,编译运行其自带的演示程序时就可能出现如下图的,105 个错误,几十个警 告的问题。 我们查看输出窗口,如下图。会发现它居然显示VC98 之类的东西,就是说它并 没有去自己的include 文件夹 中查找文件。我们可以怀疑是系统环境变量的问题了。 击Qt Creator 界面左侧的projects 图标,查看工程信息。这里我们主要查看 编辑环境Buid Environment,击其右侧的show Details。 可以看到其中的include 和lib 均指向了virtual studio 文件夹中,我们需要 将其改正。 将他们都改为自己Qt Creator 安装目录下的相关路径,如下图。(要换成你的 安装路径) 改完后会发现新的设置已经显示出来了。 我们查看下面的Run Environment,发现它已经自己改过来了。 回到编辑界面,右击工程文件,在弹出的菜单上选择Clean project,清空以前 的编译信息。 然后运行Run qmake,生成Makefile 文件。 最后,击run 或者build 都可,这时程序已经能正常编译运行了。 基于Qt 4.6 的Qt Creator 1.3.0 写helloworld 程序注意事项(原创) 注意:下面指的是在windows 下,linux 下的情况可进行相应改变 昨天Qt 4.6 和Qt Creator 1.3.0 正式版发布了,但是如果以前用过旧版本,就 可能出一些问题。 1.用debug 方式 如果你以前用了Qt 4.5 的Qt Creator,并且将QtCored4.dll,QtGuid4.dll, mingwm10.dll 等文件放到了C 盘的system 文件夹下。那么请先将它们删除,不 然编译不会通过。 编译完helloworld 程序后,如果要直接执行exe 文件,需要将安装目录(新版 Qt)下的qt/bin 目录下的QtCored4.dll,QtGuid4.dll,mingwm10.dll,和 libgcc_s_dw2-1.dll(这个是新增的)文件放在exe 文件夹中。或者将它们放到 系统的system 文件夹下。 2.选择release 方式 编译程序后生成exe 文件 1.需要Qt 安装目录下的qt/bin 目录中的QtGui4.dll ,Qt Core4.dll, libgcc_s_dw2-1.dll 以及mingwm10.dll 四个文件的支持,将它们拷贝到exe 文 件目录下。 2.程序中默认只支持png 图片,如果使用了gif,jpg 等格式的文件是显示不出 来的。需要将Qt 安装目录下的qt/plugins/目录中的imageformats 文件夹拷贝 到exe 文件目录下(注意是整个文件夹)。而imageformats 文件夹中只需要保 留你需要的文件,例如你只需要支持gif 文件,就只保留qgif4.dll 即可。 ‘Qt Creator 发布release 软件相关注意事项(原创) 注意:环境是windows 选择release 编译程序后生成exe 文件 1.需要Qt 安装目录下的qt/bin 目录中的QtGui4.dll 和 Qt Core4.dll 以及 mingwm10.dll 三个文件的支持,将它们拷贝到exe 文件目录下。 2.程序中默认只支持png 图片,如果使用了gif,jpg 等格式的文件是显示不出 来的。需要将Qt 安装目录下的qt/plugins/目录中的imageformats 文件夹拷贝 到exe 文件目录下(注意是整个文件夹)。而imageformats 文件夹中只需要保 留你需要的文件,例如你只需要支持gif 文件,就只保留qgif4.dll 即可。 Qt Creator 的 error: collect2: ld returned 1 exit status 问题 利用Qt Creator 1.2.1( Built on Sep 30 2009 at 05:21:42)编译 程序经常会出现error: collect2: ld returned 1 exit status 的错误,但是 自己的程序没有一问题,怎么回事呢? 如果这时退出软件,再重新进入,打开刚才的工程,重新编译, 就不会出现刚才的错误了。这应该是Qt Creator 软件的问题吧! 后来发现是因为上次执行的程序还在运行,你打开windows 的任 务管理器中的进程可以看见你刚才运行的程序还在执行,我们看不见,是因为它 在后台执行着。出现这个现象,是因为你写的代码的问题,比如在main 函数里 用了w.show();语句,就可能出现界面一闪而过,但它并没有关闭,而是在后台 运行,所以再次运行时就会出错。我们可以在资源管理器中将该进程关闭,或者 像上面那样直接关闭Qt Creator。 示例: #include #include "widget.h" #include "logindlg.h" int main(int argc, char *argv[]) { QApplication a(argc, argv); loginDlg m; if(m.exec()==QDialog::Accepted) { Widget w; w.show(); } return a.exec(); } 执行后就会在后台运行。这时如果修改了代码再次运行程序,就会出现上面的错 误。 在任务管理器中可以看见自己的程序: 将该进程结束,然后在重新运行,就不会出错了。 正确的代码应该这样写: int main(int argc, char *argv[]) { QApplication a(argc, argv); loginDlg m; Widget w; if(m.exec()==QDialog::Accepted) { w.show(); return a.exec(); } else return 0; //关闭整个程序 } 这样新建的对象w 就不是局部变量了,这样运行程序w 表示的窗口不会一闪而过, 会一直显示。程序也不会再出现上面的错误了。 QT 常用问题解答(转) 本文是我前几天一个网友告诉我的,当时看了感觉好,就保存下来。今天再次查 看,感觉有必要把文章享给各位学习QT 的朋友,因为网上好用的QT 资源真的 好少。 1、如果在窗体关闭前自行判断是否可关闭 答:重新实现这个窗体的closeEvent()函数,加入判断操作 Quote: void MainWindow::closeEvent(QCloseEvent *event) { if (maybeSave()) { writeSettings(); event->accept(); } else { event->ignore(); } } 2、如何用打开和保存文件对话 答:使用QFileDialog Quote: QString fileName = QFileDialog::getOpenFileName(this); if (!fileName.isEmpty()) { loadFile(fileName); } Quote: QString fileName = QFileDialog::getSaveFileName(this); if (fileName.isEmpty()) { return false; } 3、如果创建Actions(可在菜单和工具栏里使用这些Action) 答: Quote: newAct = new QAction(QIcon(":/images/new.png"), tr("&New"), this); newAct->setShortcut(tr("Ctrl+N")); newAct->setStatusTip(tr("Create a new file")); connect(newAct, SIGNAL(triggered()), this, SLOT(newFile())); openAct = new QAction(QIcon(":/images/open.png"), tr("&Open..."), this); openAct->setShortcut(tr("Ctrl+O")); openAct->setStatusTip(tr("Open an existing file")); connect(openAct, SIGNAL(triggered()), this, SLOT(open())); saveAct = new QAction(QIcon(":/images/save.png"), tr("&Save"), this); saveAct->setShortcut(tr("Ctrl+S")); saveAct->setStatusTip(tr("Save the document to disk")); connect(saveAct, SIGNAL(triggered()), this, SLOT(save())); saveAsAct = new QAction(tr("Save &As..."), this); saveAsAct->setStatusTip(tr("Save the document under a new name")); connect(saveAsAct, SIGNAL(triggered()), this, SLOT(saveAs())); exitAct = new QAction(tr("E&xit"), this); exitAct->setShortcut(tr("Ctrl+Q")); exitAct->setStatusTip(tr("Exit the application")); connect(exitAct, SIGNAL(triggered()), this, SLOT(close())); cutAct = new QAction(QIcon(":/images/cut.png"), tr("Cu&t"), this); cutAct->setShortcut(tr("Ctrl+X")); cutAct->setStatusTip(tr("Cut the current selection's contents to the " "clipboard")); connect(cutAct, SIGNAL(triggered()), textEdit, SLOT(cut())); copyAct = new QAction(QIcon(":/images/copy.png"), tr("&Copy"), this); copyAct->setShortcut(tr("Ctrl+C")); copyAct->setStatusTip(tr("Copy the current selection's contents to the " "clipboard")); connect(copyAct, SIGNAL(triggered()), textEdit, SLOT(copy())); pasteAct = new QAction(QIcon(":/images/paste.png"), tr("&Paste"), this); pasteAct->setShortcut(tr("Ctrl+V")); pasteAct->setStatusTip(tr("Paste the clipboard's contents into the current " "selection")); connect(pasteAct, SIGNAL(triggered()), textEdit, SLOT(paste())); aboutAct = new QAction(tr("&About"), this); aboutAct->setStatusTip(tr("Show the application's About box")); connect(aboutAct, SIGNAL(triggered()), this, SLOT(about())); aboutQtAct = new QAction(tr("About &Qt"), this); aboutQtAct->setStatusTip(tr("Show the Qt library's About box")); connect(aboutQtAct, SIGNAL(triggered()), qApp, SLOT(aboutQt())); 4、如果创建主菜单 答:采用上面的QAction 的帮助,创建主菜单 Quote: fileMenu = menuBar()->addMenu(tr("&File")); fileMenu->addAction(newAct); fileMenu->addAction(openAct); fileMenu->addAction(saveAct); fileMenu->addAction(saveAsAct); fileMenu->addSeparator(); fileMenu->addAction(exitAct); editMenu = menuBar()->addMenu(tr("&Edit")); editMenu->addAction(cutAct); editMenu->addAction(copyAct); editMenu->addAction(pasteAct); menuBar()->addSeparator(); helpMenu = menuBar()->addMenu(tr("&Help")); helpMenu->addAction(aboutAct); helpMenu->addAction(aboutQtAct); 5、如果创建工具栏 答:采用上面的QAction 的帮助,创建工具栏 Quote: fileToolBar = addToolBar(tr("File")); fileToolBar->addAction(newAct); fileToolBar->addAction(openAct); fileToolBar->addAction(saveAct); editToolBar = addToolBar(tr("Edit")); editToolBar->addAction(cutAct); editToolBar->addAction(copyAct); editToolBar->addAction(pasteAct); 6、如何使用配置文件保存配置 答:使用QSettings 类 Quote: QSettings settings("Trolltech", "Application Example"); QPoint pos = settings.value("pos", QPoint(200, 200)).toPoint(); QSize size = settings.value("size", QSize(400, 400)).toSize(); Quote: QSettings settings("Trolltech", "Application Example"); settings.setValue("pos", pos()); settings.setValue("size", size()); 7、如何使用警告、信息等对话框 答:使用QMessageBox 类的静态方法 Quote: int ret = QMessageBox::warning(this, tr("Application"), tr("The document has been modified.\n" "Do you want to save your changes?"), QMessageBox::Yes | QMessageBox::Default, QMessageBox::No, QMessageBox::Cancel | QMessageBox::Escape); if (ret == QMessageBox::Yes) return save(); else if (ret == QMessageBox::Cancel) return false; 8、如何使通用对话框中文化 答:对话框的中文化 比 如说,QColorDialog 的与文字相关的部,主要在qcolordialog.cpp 文件 中,我们可以从qcolordialog.cpp 用 lupdate 生成一个ts 文件,然后用自定 义这个ts 文件的翻译,再用lrelease 生成一个.qm 文件,当然了,主程序就要 改变要支持多国语言了, 使用这个.qm 文件就可以了。 另外,还有一个更快的方法,在源代码解开后有一个目录translations,下面 有一些.ts, .qm 文件,我们拷贝一个: Quote: cp src/translations/qt_untranslated.ts ./qt_zh_CN.ts 然 后,我们就用Linguist 打开这个qt_zh_CN.ts,进行翻译了,翻译完成后, 保存后,再用lrelease 命令生成qt_zh_CN.qm, 这样,我们把它加入到我们的 qt project 中,那些系统的对话框,菜单等等其它的默认是英文的东西就能显 示成中文了。 9、在Windows 下Qt 里为什么没有终端输出? 答:把下面的配置项加入到.pro 文件中 Quote: win32:CONFIG += console 10、Qt 4 for X11 OpenSource 版如何静态链接? 答:编译安装的时候加上-static 选项 Quote: ./configure -static //一定要加static 选项 gmake gmake install 然后,在Makefile 文件中加 static 选项或者在.pro 文件中加上QMAKE_LFLAGS += -static,就可以连接静态库了。 11、想在源代码中直接使用中文,而不使用tr()函数进行转换,怎么办? 答:在main 函数中加入下面三条语句,但并不提倡 Quote: QTextCodec::setCodecForLocale(QTextCodec::codecForName("UTF-8")); QTextCodec::setCodecForCStrings(QTextCodec::codecForName("UTF-8")); QTextCodec::setCodecForTr(QTextCodec::codecForName("UTF-8")); 或者 Quote: QTextCodec::setCodecForLocale(QTextCodec::codecForName("GBK")); QTextCodec::setCodecForCStrings(QTextCodec::codecForName("GBK")); QTextCodec::setCodecForTr(QTextCodec::codecForName("GBK")); 使用GBK 还是使用UTF-8,依源文件中汉字使用的内码而定 这样,就可在源文件中直接使用中文,比如: Quote: QMessageBox::information(NULL, "信息", "关于本软件的演示信息", QMessageBox::Ok, QMessageBox::NoButtons); 12、为什么将开发的使用数据库的程序发布到其它机器就连接不上数据库? 答:这是由于程序找不到数据库插件而致,可照如下解决方法: 在main 函数中加入下面语句: Quote: QApplication::addLibraryPath(strPluginsPath"); strPluginsPath 是插件所在目录,比如此目录为/myapplication/plugins 则将需要的sql 驱动,比如qsqlmysql.dll, qsqlodbc.dll 或对应的.so 文件放 到 /myapplication/plugins/sqldrivers/ 目录下面就行了 这是一种解决方法,还有一种通用的解决方法,即在可执行文件目录下写 qt.conf 文件,把系统相关的一些目录配置写到qt.conf 文件里,详细情况情参 考Qt Document Reference 里的qt.conf 部 13、如何创建QT 使用的DLL(.so)以及如何使用此DLL(.so) 答:创建DLL 时其工程使用lib 模板 Quote: TEMPLATE=lib 而源文件则和使用普通的源文件一样,注意把头文件和源文件开,因为在其它 程序使用此DLL 时需要此头文件 在使用此DLL 时,则在此工程源文件中引入DLL 头文件,并在.pro 文件中加入 下面配置项: Quote: LIBS += -Lyourdlllibpath -lyourdlllibname Windows 下和Linux 下同样(Windows 下生成的DLL 文件名为yourdlllibname.dll 而在Linux 下生成的为libyourdlllibname.so。注意,关于DLL 程序的写法, 遵从各平台级编译器所定的规则。 14、如何启动一个外部程序 答:1、使用QProcess::startDetached()方法,启动外部程序后立即返回; 2、使用QProcess::execute(),不过使用此方法时程序会最阻塞直到此方法执 行的程序结束后返回
NEAT 开 发 指南 文档 适用于 PT80 系列 移动数据终端 版本记录 版本号 版本描述 发布日期 V 1.0 初始版本。 2012-04-12 V1.1 修改前三章内容 2012-09-25 目录 第一章 关于本手册........................................................................................................................................ 1 简介 ........................................................................................................................................................ 1 相关文档 ................................................................................................................................................. 1 章节介绍 ................................................................................................................................................. 1 版权和许可条款 ...................................................................................................................................... 1 第二章 PT80 开发入门 .................................................................................................................................. 2 开发环境搭建 ......................................................................................................................................... 2 使用 NEAT 工程向导建立应用程序 ........................................................................................................ 5 编译及运行程序(模拟器下) ................................................................................................................ 7 编译及运行程序(PT80) .................................................................................................................... 11 下载 PT80 应用程序 ............................................................................................................................. 12 第三章 PT80 NEAT 编程基础 ..................................................................................................................... 17 事件驱动和消息响应机制 ..................................................................................................................... 17 建立一个应用程序 ................................................................................................................................ 17 应用程序的关闭 .................................................................................................................................... 19 框架窗口 ............................................................................................................................................... 19 完整的例子 ........................................................................................................................................... 20 NEAT 程序一般执行过程 ..................................................................................................................... 20 第四章 窗口 ................................................................................................................................................ 21 窗口的概念 ........................................................................................................................................... 21 窗口的创建和删除 ................................................................................................................................ 22 窗口类型 ............................................................................................................................................... 23 窗口事件 ............................................................................................................................................... 24 窗口类概览 ........................................................................................................................................... 24 基础窗口类 ........................................................................................................................................... 26 窗口类 CNeatWnd ................................................................................................................................ 26 窗口类 CNeatView 视图类 ................................................................................................................... 26 CNeatControl 类 ................................................................................................................................... 26 CNeatFrame 类 .................................................................................................................................... 26 第五章 消息与消息处理 .............................................................................................................................. 27 消息驱动的编程模型 ............................................................................................................................ 27 消息及消息处理过程 ............................................................................................................................ 27 第六章 在窗口中绘画 .................................................................................................................................. 30 设备上下文 ........................................................................................................................................... 30 绘画工具 ............................................................................................................................................... 32 绘制基本图形 ....................................................................................................................................... 34 绘制文本 ............................................................................................................................................... 34 第七章 处理用户输入 .................................................................................................................................. 36 PT80 的按键所对应的键值 .................................................................................................................. 36 输入事件相应 ....................................................................................................................................... 36 第八章 对话框编程基础 .............................................................................................................................. 37 模态和非模态对话框编程 ..................................................................................................................... 37 通用对话框 ........................................................................................................................................... 38 对话框示例 ........................................................................................................................................... 38 对话框资源 ........................................................................................................................................... 40 第九章 NEAT 控件 ...................................................................................................................................... 41 控件综述 ............................................................................................................................................... 41 静态框 .................................................................................................................................................. 41 编程示例 ............................................................................................................................................... 42 按钮 ...................................................................................................................................................... 43 列表框 .................................................................................................................................................. 45 组合框 .................................................................................................................................................. 46 编辑框 .................................................................................................................................................. 48 进度条 .................................................................................................................................................. 50 滑块 ...................................................................................................................................................... 51 旋钮 ...................................................................................................................................................... 52 第十章 资源及资源模板 .............................................................................................................................. 53 概述 ...................................................................................................................................................... 53 图标 ...................................................................................................................................................... 53 对话框 .................................................................................................................................................. 54 菜单 ...................................................................................................................................................... 54 第十一章 编写国际化程序 ........................................................................................................................... 57 国际化简介 ........................................................................................................................................... 57 如何实现国际化 .................................................................................................................................... 57 使用 NEAT 平台开发国际化应用程序 .................................................................................................. 59 小结 ...................................................................................................................................................... 61 第十二章 编写多线程程序 ........................................................................................................................... 62 多线程简介 ........................................................................................................................................... 62 如何使用 wxThread 线程类 .................................................................................................................. 62 线程同步对象 ....................................................................................................................................... 64 编程实例 ............................................................................................................................................... 69 小结 ...................................................................................................................................................... 69 第十三章 网络编程...................................................................................................................................... 70 使用 wxSocket 编程 ............................................................................................................................. 70 Socket 类和功能概览 ........................................................................................................................... 70 Socket 及其基本处理介绍 .................................................................................................................... 70 Socket 标记 .......................................................................................................................................... 75 使用 Socket 流 ..................................................................................................................................... 78 第十四章 数据库编程 .................................................................................................................................. 81 wxSqlite3 简介 ..................................................................................................................................... 81 如何使用 wxSqlite3 .............................................................................................................................. 81 编程实例 ............................................................................................................................................... 92 小结 ...................................................................................................................................................... 92 第十五章 设备编程...................................................................................................................................... 93 PT80 设备操作类 ................................................................................................................................. 93 PT88WIFI 操作类 ................................................................................................................................. 93 PT88BT 操作类 .................................................................................................................................... 93 PT88 配置信息操作类 .......................................................................................................................... 93 第十六章 wxBase 编程接口 ........................................................................................................................ 94 概述 ...................................................................................................................................................... 94 时间日期 ............................................................................................................................................... 94 动态库 .................................................................................................................................................. 94 字符串 .................................................................................................................................................. 94 文件、文件夹、流 ................................................................................................................................ 95 附录 1 系统调用 .......................................................................................................................................... 97 1 第一章 关于本手册 简介 NEAT 是 Newland Embedded Application Toolkit 的缩写,由福建新大陆电脑股份有限公司开发,其目标是为开发者提供一套可 靠、高效、易用的跨平台应用开发支撑系统。PT80 使用 NEAT 这个软件做为 PT80 应用程序的开发工具,开发人员通过 NEAT 可 以快速开发出在 PT80 上运行的应用程序。 本手册详细讲述了利用 NEAT 的基础知识、技术资料和开发技巧,以及 NEAT 当前。 相关文档 章节介绍 版权和许可条款 2 第二章 PT80 开发 入门 开发环境搭建 运行环境 NEAT 做为 Microsoft 开发工具 Visual Studio 2005 的插件存在, 开发人员首先要安装 Microsoft Visual Studio 2005, 当前版本 NEAT 只 v0.2.0 只支持 Visual Studio 2005。开发人员需要使用 Visual Studio 中的 Visual C++ 进行 NEAT 应用程序的开发。 NEAT 简介 NEAT 系统大概包含以下几部内容:  各种库及头文件  交叉编译工具及各种辅助工具  Windows 端的模拟器  帮助文档及示例代码 NEAT 安装步骤 运行安装包,弹出如下界面: 3 \image html chapter2_setup1.jpg 击―下一步‖,进入安装准备状态: \image html chapter2_setup2.jpg 击安装,即进入安装过程,安装程序将在 D:\NEAT(默认安装路径,可修改)下安装所有内容。 \image html chapter2_setup3.jpg 安装完成后,有以下界面: 4 \image html chapter2_setup4.jpg 击完成即可。安装程序将生成 NEAT 开发向导,NEAT 开发手册,NEAT 模拟器的快捷方式。 5 使用 NEAT 工程向导建立应用程序 使用 NEAT ―工程向导‖(NeatProjectWiz)来快速建立 NEAT 应用程序。启动方法:击“开始”、“程序”;选择“Neat 嵌 入式应用开发套件”中的“工程向导”(如下图)。 NEAT 工程向导使用步骤: 1. 设置开发环境,目前只支持 VS2005 2. 设置平台,目前只支持 PT80 3. 设置程序类型,为两大类:  应用程序:即当前 PT80 系统菜单里头的应用程序,应用程序是做为系统菜单的插件而运行。  系统程序:类似 PT80 的整体程序(当前 NEAT 版本不支持,保留功能,新建工程时请勿选择此模式)。 注:每类又包含两种模式,及对话框模式和非对话框模式。  非对话框模式:开发人员需要用代码确定图形控件的放置位置和大小。 6  对话框模式:开发人员可以将图形控件从 Visual Studio 中‖Control Toolbar‖中拖放到 PT80 可视化对话框界面,不 需要使用代码确定图形界面的放置位置和大小,推荐开发人员新建应用程序时使用此模式。 \image html chapter2_projWiz1.jpg 4. 填写工程名称及工程所在的路径。 \image html chapter2_projWiz2.jpg 7 5. 击确定,NEAT 工程向导会开始创建 VC 工程。 6. 工程创建后会提示是否要打开新建工程,击确定打开(如下图) 注意:1. 如果击‘ 是’ 后打不开工程 , 请设置在 在 Windows 操作系统中设置 VS2005 的环境变量 ,即将 VS2005 的启动路径添 加到系统的”Path” 环境变量中 。 2. 如果在工程目录下已经存在工程名和新生成的工程名相同的工程时,会被新的工程覆盖。 编译及运行程序(模拟器下) 使用 NEAT “ 工程向导” (NeatProjectWiz )来快速建 编译和运行 应用程序 创建完成 PT80 应用程序工程后,可以参考下列编译和运行应用程序: Step1: 使用 VC 打开已建立好的 NEAT VC 工程。 Step2: 打开当前工程的配置管理器(如下图) 8 9 针对当前项目,―配置‖选择―WIN32‖,―平台‖选择―Win32‖。如果你想脱离主控系统程序,独立运行应用程序,请在―配置‖中 选择―WIN32-Alone‖。 \image html chapter2_appProjCfg.jpg “配置管理器的配置” 检查工程属性的―调试‖项,在―命令‖栏中填写―d:\neat\pt880system.exe‖(默认安装路径)。 \image html chapter2_appProjDebugCfg.jpg “调试的配置” 编译并运行程序。PT80 模拟器将自动被打开,脱离主控系统程序运行的应用程序,接下来请直接看 Step4。 Step4: (1)在模拟器上,选择―系统菜单‖的―系统设置‖项,进入―系统设置‖界面,选择"1.设定程序"项; 10 选择"系统设置"项进入 选择"设定程序"项进入 (2)进入"设定程序"项后,在插件栏按"OK"键显示可选插件后,选择要运行的程序。然后在"开机运行"项设置为"是"; 选择好需要运行的程序后 将"开机自动运行"设为"是" (3)选择设定后会自动回到"系统设置"界面。按"ESC"退出到"系统菜单"界面后选择"1 运行程序"项即可; 选择"运行程序"项后即运行已设定的程序 (4)下一次编辑运行时即自动进入程序。 11 应用程序成功启动。你可以击模拟器上的按钮进行操作,你也可以在程序中添加断等工具,进行调试。 \image html chapter2_appProjDebuging.jpg “调试应用程序” 编译程序(PT80 ) 在模拟器上编译运行PT80应用程序正常后, 需要使用NEAT编译工具编译在PT80应用程序。 击NEAT工具栏上的―Compile‖ 按钮,如果没找到 NEAT 工具栏,请右键当前工具栏空白区域,选择―NEAT‖。成功编译后,可以看到―输出‖窗口中关于.so 文件或 可执行文件的信息。 \image html chapter2_NEATtoolbar.jpg “NEAT 工具栏” 12 \image html chapter2_NEAToutput.jpg “成功编译后输出窗口的内容” 下载 PT80 应用程序 在编译完成 PT80 应用程序后,有两种方法将编译完成的 PT80 应用程序下载到 PT80 上:“USB 模式下载”和“NEAT 插件 下载”。 USB 模式下载应用程序 Step1: 在 PT80 上将 USB 数据通讯方式设置为“U 盘”(具体设置方式可参考《PT80 用户手册》)。 Step2: 将 PT80 和 PC 用 USB 线缆连接 Step3: 将编译完成的 *.so 文件复制到 U 盘下的“app”目录中(如下图): 13 Step4: 在 PT80 的系统菜单里选择刚下载的应用程序运行(运行以下载应用程序的方法可参考《PT80 用户手册》)。 使用 NEAT 插件工具下载应用程序 Step1: 在 PT80 上将 USB 数据通讯方式设置为“串口下载模式”(具体设置方式可参考《PT80 用户手册》)。 Step2: 将 PT80 和 PC 用 USB 线缆连接 Step3: 在 PC 端安装“USB 转串口”驱动(PT80 USB 转串口驱动请访问新大陆自动识别网站 www.nlscan.com 上下载) Step4: 驱动安装完成后,在 PC 上会新增一个虚拟串口,在 PC 上的“设备管理器”中可以查看虚拟串口号(如下图),记录 下这虚拟串口号。 14 Step5: 击“下载”按键(如下图) 在下载对话框中(入下图)按照以下步骤下载 PT80 应用程序到 PT80  选择正确的虚拟串口  选择需要下载 PT80 应用程序,应用程序在 VC 工程目录下的“NEAT_OBJ”目录,应用程序是以“.so”为后缀名 的文件  击“下载”按键 15 下载成功后,在下载对话框上会提示下载成功(如下图): 16 Step6: 至此下载 PT80 应用程序完成, 可以在 PT80 上开始运行应用程序 (具体运行应用程序的方法可参考 《PT80 用户手册》 ) 。 17 第三章 PT80 NEAT 编程基础 事件驱动和消息响应机制 NEAT 程序设计是一种事件驱动的程序设计模式,在程序提供给用户的界面中有许多可操作的可视对象。用户可以从所有可 能的操作中任意选择,被选择的操作会产生某些特定的事件,这些事件发生后的结果是向程序中的某些对象发出消息,然后这些对 象调用相应的消息处理函数来完成特定的操作。NEAT 应用程序最大的特就是程序没有固定的流程,而只是针对某个事件处理有 特定的子流程,NEAT 应用程序是由许多这样的子流程构成的。 NEAT 应用程序是面向对象的。程序提供给用户界面的可视对象在程序的内部一般也被看成一个对象,用户对可视对象的操 作通过事件驱动模型触发相应的消息处理函数。 程序的运行过程就是用户的外部操作不断产生事件, 这些事件又不断被处理的过程。 NEAT 这种事件驱动模型源于消息响应机制。在 NEAT 系统中,事件产生消息,消息对应事件,所谓事件响应,其实就是对 各种消息的响应。NEAT 系统会不断的捕捉各种消息,并把捕捉到的消息发送到应用程序,应用程序将消息再传递给相关的消息处 理函数做相应的处理。这种等待消息、响应消息的操作方式就是 NEAT 的消息处理机制,类似于 Windows 的消息处理机制。 下面是 NEAT 应用程序的工作原理示意图。 \image html neat-message.jpg "NEAT 消息处理机制" 建立 一个应用程序 每一个 NEAT 程序都需要定义一个 \ref CNeatApp 类的派生类,并需要且只能构造一个这个类的实例,这个实例控制着整个 程序的执行。你的这个继承自 \ref CNeatApp 的子类至少需要定义一个 OnInit 函数,当 NEAT 准备好运行你写的代码的时候,它 将会调用这个函数(和一个典型的 Win32 程序中的 main 函数或者 WinMain 函数类似)。 你定义这个子类的代码可能和下面的代码类似: \code class MyApp : public CNeatApp { public: virtual bool OnInit(int args, const char* arg[]); 18 virtual int OnExit(); }; \endcode 在这个 OnInit 函数中,你通常应该创建至少一个窗口,对传入的命令行参数进行解析,为应用程序进行数据设置和其它的一 些初始化的操作。然后 NEAT 将开始消息循环,用来处理用户输入并且在必要的情况下处理这些输入。如果 OnInit 函数返回假, NEAT 将会释放它内部已经配的资源,然后结束整个程序的运行。 接下来我们看一个最简单的 OnInit 函数的实现: \code CMyFrame m_FrameWnd; // 应用程序初始化函数 bool CMyApp::OnInit(int args, const char* arg[]) { \code CMyFrame m_FrameWnd; // Initialization function of the application program bool CMyApp::OnInit(int args, const char* arg[]) { // 调用 Create 函数来创建主框架窗口 m_FrameWnd.Create(_("Minimal"), WS_THINFRAME|WS_CAPTION ); // Call the Create function to create the frame window m_FrameWnd.Create(_("Minimal"), WS_THINFRAME|WS_CAPTION ); // 显示主窗口 m_FrameWnd.ShowWindow(SW_SHOW); return true; } // Show the window m_FrameWnd.ShowWindow(SW_SHOW); return true; } \endcode \endcode 你可能还会注意到上面例子中的_()这个宏,在接下来的例子中,这个宏还会被频繁用到。它的作用是用来告诉 NEAT 将其 中的字符串翻译成其它语言的版本,参见―编写国际化程序‖。 那么创建 MyApp 实例的代码在哪里呢?实际上, 这是在 NEAT 内部实现的, 不过你仍然需要告诉 NEAT 需要创建哪一个 App 类的实例,所以你还需要增加下面的一个宏: \code IMPLEMENT_APP(MyApp) \endcode 如果没有实现这个类,NEAT 就不知道怎样创建一个新的应用程序对象。这个宏除了上述的功能以外,还会检查编译应用程 序使用的库文件是否和当前的库文件的版本相匹配,如果没有这种检查,由此而产生的一些运行期的错误可能很难被查出原因。 19 应用程序的关闭 当框架窗口关闭后,应用程序也跟着关闭了,在应用程序关闭前,它会调用 \ref CNeatApp::OnExit 函数,可以在这里增加 应用程序退出前的操作。 举个例子: \code class MyApp : public CNeatApp { public: CNeatWnd *m_helpCtrl; ... }; bool MyApp::OnInit(int argc,const char* argv[]) { ... m_helpCtrl = new CNeatWnd; ... return true; } int MyApp::OnExit() { delete m_helpCtrl; return 0; } \endcode 框架窗口 我们来看一看自定义的派生至 CNeatFrameWnd 窗口类的 MyFrame。一个 \ref CNeatFrameWnd 窗口是一个可以容纳别的 窗口的顶级窗口,通常拥有一个标题栏和一个状态栏和一个客户视图。下面是我们的例子中这个类的定义,可以将其放在 MyApp 的定义之前: \code // 从 CNeatFrameWnd 派生一个框架窗口类,做为该应用程序的框架窗口 class CMyFrame : public CNeatFrameWnd { public: // 窗口刷新事件的处理函数 virtual int OnPaint(); }; \endcode \code // Derive a frame window class from CNeatFrameWnd and use it as the frame window for the application program class CMyFrame : public CNeatFrameWnd { 20 public: // handler function for the window refreshing events virtual int OnPaint(); }; \endcode 这个窗口类的中只定义了一个用来把窗口刷新事件的处理函数。 完整的例子 现在把所有的代码放在一起了,通常,我们应该把头文件和实现文件开,但是对于这样小的一个程序,就没有这个必要了。 \include minimal\minimal.cpp \image html demo-minimal.jpg "NEAT 最小应用程序" NEAT 程序一般执行过程 下面大概描述一下整个程序的执行过程: 1. 程序执行时,main 函数被调用,NEAT 初始化它自己的数据结构并且创建一个 MyApp 的实例。 2. NEAT 调用 MyApp::OnInit 函数, 这个函数会创建一个 MyFrame 的实例,MyFrame 通过调用 Create 来创建一个窗口, MyApp::OnInit 函数显示主窗口并且返回真。 3. NEAT 开始进入消息循环,等待事件发生并且将事件发给相应的处理过程。 4. 应用程序会在以下情况下退出:主窗口被关闭,用户选择退出菜单或者系统按钮和系统菜单中的关闭选项(这些系统菜 单和系统按钮在不同的系统中就往往千差万别了)。 21 第四章 窗口 你当然大略的知道一个窗口指的是什么,但是为了更好的理解 NEAT 窗口相关的 API,你应该更精通 NEAT 所使用的窗口模 型的细节。它可能和你在某个特定平台上的窗口概念有些许的不同。下图演示了一个窗口中的各个基本元素: \image html neat-window.jpg "NEAT 窗口" 窗口的概念 一个窗口指的是屏幕上的任何一个拥有以下特征的规则区域:它可以被改变大小,可以自我刷新,可以被显示和隐藏等等。 它可以包含别的窗口(比如 frame 窗口就可以包含菜单条窗口,工具条窗口以及状态条窗口),也可以子窗口(比如一个静态的文本或 者一副静态图片)。通常你在使用 NEAT 编写的程序运行的屏幕上看到的窗口,都和一个 \ref CNeatWnd 类或者它的派生类对应。 客户区和非客户区 当我们谈到窗口的大小,我们通常指的是它整个的大小,包括一些用于修饰的边框和标题栏等。而当我们谈到一个窗口的客 22 户区大小,通常都只意味着窗口里面那些能被绘制或者它的子窗口能被放置的位置的大小。例如一个 frame 窗口的客户区大小就不 包括那些菜单栏,状态栏和工具栏所占用的地方。 滚动条 大多数窗口都有显示滚动条的能力,这些滚动条通常是窗口自己增加的而不是由应用程序手动增加的。在这种情况下,客户 区的大小还应该减去滚动条所占用的空间。 为了优化性能, 只有那些拥有 WS_HSCROLL 和 WS_VSCROLL 类型的窗口才会自动生 成它们自己的滚动条。 座标体系 窗口的座标体系通常是左上角为原(0,0),单位是象素。 窗口绘制 当一个窗口需要重绘的时候,它将收到两个事件,MSG_ERASEBKGND 事件用于通知应用程序重新绘制背景,对应的消息处 理函数为OnEraseBkgnd, MSG_PAINT则用于通知重新绘制前景, 对应的消息处理函数为OnPaint。 那些常用控件比如CNeatButton(按 钮)已经处理这两个事件,但是如果你是要创建自己的窗口控件,你就需要自己处理这两个事件。通过获取窗口的变动区域你可以 优化你的绘制代码。 颜色和字体 每一个窗口都有一个前景色和一个背景色。默认的背景擦除函数会使用背景色来清除窗口背景,如果没有设置背景色,则会 使用系统默认的背景颜色进行背景的清除。前景色为文本输出的字体颜色。每一个窗口也拥有一个字体设置,是否用到这个字体设 置要取决于这个窗口本身的类型。 改变大小 当一个窗口的大小,无论是来自用户还是应用程序本身的原因,发生变化时,它将收到一个 MSG_SIZE 事件,对应的消息处理 函数为 OnSize。 输入 只有当前处于活动状态的窗口才可以接收键盘事件。应用程序自己可以设置自己为活动状态,NEAT 也会在用户击某个窗 口的时候将其设置为活动状态。正变成活动状态的窗口会收到 MSG_SETFOCUS 事件,对应的消息处理函数为 OnSetFocus,而正 失去焦的窗口会收到 MSG_KILLFOCUS 事件,对应的消息处理函数为 OnKillFocus。 窗口的创建和删除 大多数的窗口类都可以需要两步来创建,首先定义一个窗口类对象,然后调用 Create 函数来创建窗口。下面演示一个创建窗 口的代码片段: \code CNeatWnd *mywnd = new CNeatWnd; 23 mywnd->Create( ―mywindow‖, WS_CHILD, 10, 10, 100, 100,parent); \endcode 你可以传递一个字符串的名字,一个类型 (接下来会提到),位置和大小参数给这个窗口。除非是 frame 或者 dialog 窗口,对 于别的窗口, 都必须在 Create 函数中传入一个非空的父窗口, 这会把这个新窗口作为这个父窗口的子窗口, 当父窗口被释放的时候, 它的所有的子窗口也将被释放。 窗口在你调用 Create 函数的时候会收到 MSG_CREATE 事件,对应的消息函数为 OnCreate,你可以对这个事件进行进一步的 处理。 当你创建一个窗口类,或者其它任何非顶层窗口的派生类的时候,如果它的父窗口是可见的,那么它也总是可见的,你可以 通过 ShowWindow(SW_HIDE)来使它不可见或使用 ShowWindow(SW_SHOW)来使它可见。 你可以通过向窗口发送 MSG_CLOSE 消息(对应的消息处理函数为 OnClose)来关闭窗口。通过调用 DestroyWindow 函数来释 放窗口的资源,MSG_DESTROY 事件(对应的消息函数为 OnDestroy)会在窗口刚刚要被释放之前被调用。 窗口拥有一个类型和一个扩展类型。窗口类型是设置窗口创建时的行为和外观的一种简洁的方法。这些类型的值被设置成可 以使用类似比特位的方法操作,例如下面的例子: WS_CAPTION | WS_THICKFRAME|WS_VISIBLE CNeatWnd 类有一组基本的类型值,例如边框的类型等,每一个派生类可以增加它们自己的类型。需要特别指出的是,扩展类 型的值是不可以拿来给类型用的。 窗口类型 每一个窗口类都可以使用定义在下表中的这些的窗口类型。这些类型中不是所有的类些都被所有的控件所支持。需要注意的 是以 WS_开头的类型用于 dwStyle 的设置,以 WS_EX_开头的类型用于 dwExStyle 的设置,两个不能互用。 通用窗口类型: 风格标识 含义 备注 WS_VISIBLE 创建初始可见的窗口 WS_DISABLED 创建初始被禁止的窗口 WS_CAPTION 创建含标题栏的主窗口 仅用于主窗口 WS_SYSMENU 创建含系统菜单的主窗口 仅用于主窗口 WS_BORDER 在窗口周围显示一个边框 WS_THICKFRAME 创建具有厚边框的窗口 WS_THINFRAME 创建具有薄边框的窗口 WS_VSCROLL 创建带垂直滚动条的窗口 WS_HSCROLL 创建带水平滚动条的窗口 24 WS_MINIMIZEBOX 标题栏上带最小化按钮 仅用于主窗口 WS_MAXIMIZEBOX 标题栏上带最大化按钮 仅用于主窗口 WS_EX_TRANSPARENT 透明窗口风格 仅用于部控件,如编辑框和滚动 窗口控件等 WS_EX_NOCLOSEBOX 主窗口标题栏上不带关闭按钮 窗口事件 窗口类和它的派生类可以产生下面的事件,在窗口里有对应的事件处理函数,所有的事件处理函数的返回类型为 int,如果返 回值为 0 表示该事件处理函数返回后,继续执行 NEAT 对该事件的默认处理,如果返回值为非 0,表示不再执行系统的默认处理。 通用窗口事件及消息处理函数: 窗口事件 消息处理函数 备注 MSG_CREATE \ref CNeatWnd::OnCreate 窗口创建事件 MSG_CLOSE \ref CNeatWnd::OnClose 窗口关闭事件 MSG_DESTROY \ref CNeatWnd::OnDestroy 窗口销毁事件 MSG_ERASEBKGND \ref CNeatWnd::OnEraseBkgnd 窗口背景擦除事件 MSG_PAINT \ref CNeatWnd::OnPaint 窗口客户区刷新 MSG_KEYDOWN \ref CNeatWnd::OnKeyDown 按键按下事件 MSG_KEYUP \ref CNeatWnd::OnKeyUp 按键释放事件 MSG_CHAR \ref CNeatWnd::OnChar 字符事件 MSG_TIMER \ref CNeatWnd::OnTimer 定时器事件 MSG_COMMAND \ref CNeatWnd::OnCommand 命令事件 MSG_SETFOCUS \ref CNeatWnd::OnSetFocus 窗口获得焦事件 MSG_KILLFOCUS \ref CNeatWnd::OnKillFocus 窗口失去焦事件 MSG_SIZE \ref CNeatWnd::OnSize 窗口大小调整事件 MSG_SHOWWINDOW \ref CNeatWnd::OnShowWindow 窗口显示(不显示)事件 MSG_HSCROLL \ref CNeatWnd::OnHScroll 水平滚动事件 MSG_VSCROLL \ref CNeatWnd::OnVScroll 垂直滚动事件 MSG_ENABLE \ref CNeatWnd::OnEnable 窗口允许事件 MSG_IDLE \ref CNeatWnd::OnIdle 窗口进入 IDEL 事件 窗口类概览 在接下来的章节中,我们会介绍最常用的那些窗口类以便你可以在你的应用程序中使用它们。 25 基本窗口类 下面的这些基本的窗口类实现了一些最基本的功能,这些类主要是用来作为别的类型的基类以生成更实用的派生类。 窗口类 描述 \ref CNeatWnd 这是所有窗口类的基类。 \ref CNeatControl 所有控件(比如 CNeatButton)的基类。 顶层窗口类 顶层窗口类通常指那些独立的位于桌面上的类。 窗口类 描述 \ref CNeatFrame 一个可以包含其他窗口,并且大小可变的窗口类。 \ref CNeatDialog 是一种可变大小的用于给用户提供选项的对话框窗口类。 视图类 窗口类 描述 \ref CNeatView 这是所有视图类的基类。 \ref CNeatMenuView 一个实现菜单选择功能的视图,支持文本和图标模式。 \ref CNeatTreeView 一个实现树型功能的视图。 \ref CNeatListView 一个实现列表功能的视图。 控件窗口类 这些控件是用户可以操作或者编辑的。 窗口类 描述 \ref CNeatStatic 静态框 \ref CNeatButton 按钮 \ref CNeatEdit 编辑框 \ref CNeatProgressCtrl 进度条 \ref CNeatListBox 列表框 \ref CNeatComboBox 组合框 \ref CNeatScrollBar 滚动条 \ref CNeatMonthCalendar 日历 \ref CNeatSliderCtrl 滑块 \ref CNeatSpinButtonCtrl 旋钮 26 基础窗口类 虽然你不一定有机会直接使用基础窗口类(CNeatWnd),但是由于这个类是很多窗口控件的基类,它实现的很多方法在它的子 类型中都可以直接拿来使用,所以有必要介绍一下这个基础窗口类。 窗口类 CNeatWnd \ref CNeatWnd 窗口类既是一个重要的基类, 也是一个你可以直接在代码中使用的类。 当然, 前者使用的频度要比后者大很多。 CNeatWnd 类的成员函数。 因为 CNeatWnd 类是其它所有窗口类的基类,它拥有很多的成员函数。我们不在这里作一一的说明,只能拣其中最重要的一 些作简要的说明。具体的内容参见\ref CNeatWnd 的接口说明,以便能够彻底了解 CNeatWnd 类提供的所有功能,以及要使用这个 功能你需要提供的参数等。 TODO:增加部成员函数的使用介绍。 窗口类 CNeatView 视图类 CNeatControl 类 \ref CNeatControl 继承自\ref CNeatWnd 类, 用来作为控件的基类,所谓控件指的是那些可以显示数据项并且通常需要响应鼠 标或者键盘事件的那些窗口类。 CNeatFrame 类 顶层窗口直接被放置在桌面上而不是包含在其它窗口之内。如果应用程序允许,他们可以被移动或者重新改变大小。有两种 基础的顶层窗口类型。 CNeatFrameWnd 和 CNeatDialog 都是从 CNeatWnd 继承来的。一个对话框既可以是模态的也可以是非模态 的,而 frame 通常都是非模态的。模态对话框的意思是说当这个对话框弹出时,应用程序除了等待用户关闭这个对话框以外不再作 别的事情。对于那些要等待用户响应以后才能继续的操作来说,这是比较合适的。 顶层窗口通常都拥有一个标题栏,这个标题栏上有一些按钮或者菜单或者别的修饰用来关闭,或者最小化,或者恢复这个窗 口。而 frame 窗口则通常还会拥有菜单条,工具条和状态条。但是通常对话框则没有这些。 27 第五章 消息与消息处理 消息驱动的编程模型 所有的 GUI 程序都是事件驱动的。换句话说,应用程序一直停留在一个循环中,等待着来自用户或者其他地方(比如窗口刷 新或网络连接)的事件,一旦收到某种事件,应用程序就将其扔给处理这个事件的函数。虽然看上去不同的窗口是同时被刷新的, 但实际上,绝大多数的 GUI 程序都是单线程的,因此窗口的刷新是依次按顺序进行的。如果由于某种意外你的设备变得很慢导致 窗口刷新的过程变的很明显,你就会注意到这一。 不同的 GUI 编程架构用不同的方法将它内部的事件处理机制展现给程序开发者。对于 NEAT 来说,消息函数重载是最主要的 方法。在下一小节我们会对此进行进一步的解释。 NEAT 应用程序通过接收消息来和外界交互。消息由系统或应用程序产生,系统对输入事件产生消息,系统对应用程序的响 应也会产生消息,应用程序可以通过产生消息来完成某个任务,或者与其它应用程序的窗口进行通讯。总而言之,NEAT 是消息驱 动的系统,一切运作都围绕着消息进行。 系统把消息发送给应用程序窗口过程,窗口过程有四个参数:窗口句柄、消息标识以及两个 32 位的消息参数。窗口句柄决定 消息所发送的目标窗口,NEAT 可以用它来确定向哪一个窗口过程发送消息。消息标识是一个整数常量,由它来标明消息的类型。 如果窗口过程接收到一条消息,它就通过消息标识来确定消息的类型以及如何处理。消息的参数对消息的内容作进一步的说明,它 的意义通常取决于消息本身,可以是一个整数、位标志或数据结构指针等。对其他不同的消息类型来讲,wParam 和 lParam 也具有 明确的定义。应用程序一般都需要检查消息参数以确定如何处理消息。 消息及消息处理过程 NEAT 事件处理系统采用通常的虚方法机制来实现。每一个 CNeatWnd 的派生类,例如 frame,按钮,对话框等,都会在其内 部重载消息处理函数,用来告诉 NEAT 事件和事件处理过程的对应关系。 要重载一个消息处理函数,你需要下面两个步骤: 1. 在派生类里声明重载的消息函数,类型要和基类中定义的一样。 2. 实现该消息处理函数。 让我们来扩展一下前一章中的例子,来增加一个按键事件的处理。下面是扩展以后的 MyFrame 的定义: \code class CMyFrame : public CNeatFrameWnd { public: // 窗口刷新事件的处理函数 virtual int OnPaint(); // 按键事件的处理函数 virtual int OnKeyDown(UINT nKeyCode, UINT nRepCnt, UINT nFlags); 28 }; \endcode 增加处理的实现部: \code // 按键按下事件的消息处理函数 int CMyFrame::OnKeyDown(UINT nKeyCode, UINT nRepCnt, UINT nFlags) { wxString str = _("OnKeyDown: "); // nKeyCode 按键的扫描码 switch (nKeyCode) { case KEY_0: str += "0"; break; case KEY_1: str += "1"; break; case KEY_2: str += "2"; break; case KEY_3: str += "3"; break; case KEY_4: str += "4"; break; case KEY_5: str += "5"; break; case KEY_6: str += "6"; break; case KEY_7: str += "7"; break; case KEY_8: str += "8"; break; case KEY_9: str += "9"; break; case KEY_UP: str += "UP"; break; case KEY_FUNC: str += "FN"; break; default: str += wxString::Format("0xX",nKeyCode); } CNeatPaintDC dc(this); str += " "; dc.TextOut(0,0,str); // 如果不要基类来处理一些默认的实现,可以在这里直接返回 // return 1; // 交给基类来完成一些默认处理 return CNeatFrameWnd::OnKeyDown(nKeyCode,nRepCnt,nFlags); } \endcode 消息循环 在每个主窗口及对话框后面都存在一个消息循环,消息循环就是一个循环体,在这个循环体中,程序利用 GetMessage 函数不 停地从消息队列中获得消息,然后利用 DispatchMessage 函数将消息发送到指定的窗口,也就是调用指定窗口的窗口过程,并传递 消息及其参数。典型的消息循环如下所示: \code MSG Msg; while ( GetMessage(&Msg, m_hWnd) ) { 29 TranslateMessage (&Msg); DispatchMessage (&Msg); } \endcode 如上所示,应用程序在创建了主窗口之后开始消息循环。 GetMessage 函数从\a m_hWnd 窗口所属的消息队列当中获得消息,然后调用 TranslateMessage 函数将击键消息 MSG_KEYDOWN 和 MSG_KEYUP 翻译成字符消息 MSG_CHAR ,最后调用 DispatchMessage 函数将消息发送到指定的窗口。 GetMessage 函数直到在消息队列中取到消息才返回,一般情况下返回非 0 值;如果取出的消息为 MSG_QUIT,GetMessage 函数 将返回 0,从而使消息循环结束。结束消息循环是关闭应用程序的第一步,应用程序一般在主窗口的窗口过程中通过调用 PostQuitMessage 来退出消息循环。 消息事件相应函数 NEAT 在 NEAT 消息处理机制之上进行了进一步的封装,它把消息循环封装在\ref CNeatApp 和\ref CNeatWnd 等基类里,从 应用程序的角度来看,它是看不到消息循环及消息派发的过程。NEAT 把每个消息的处理过程定义成消息事件响应函数,这些响应 函数大部定义在\ref CNeatWnd 中,和控件通知消息相关的响应函数定义在\ref CNeatDialog 类里面。 30 第六章 在窗口中绘画 设备上下文 理解设备上下文 在 NEAT 中,所有的绘画相关的动作,都是由设备上下文完成的。每一个设备上下文都是\ref CNeatDC 的一个派生类。每次 在窗口上绘画,都要先创建一个窗口绘画设备上下文,然后在这个上下文上绘画。 可用的设备上下文 在 NEAT 中,所有的绘画相关的动作,都是由设备上下文完成的。每一个设备上下文都是\ref CNeatDC 的一个派生类。每次 在窗口上绘画,都要先创建一个窗口绘画设备上下文,然后在这个上下文上绘画。 下面列出了你可以使用的设备上下文: \ref CNeatDC 设备上下文的基类,其他各种设备上下文都是派生自这个类. \ref CNeatClientDC 用来在一个窗口的客户区绘画。 \ref CNeatPaintDC 仅用在重绘事件的处理函数中,用来在窗口的客户区绘画。 当使用\ref CNeatDC 中的输出函数在屏幕上画图时,输出的某些特性并没有在函数调用过程中规定,它是通过设备上下文的 属性获得。 例如, 在调用 CNeatDC::DrawText 时, 要指定待输出的字符串和显示该字符串的矩形区域, 但没有指定文本颜色和字体, 因为颜色和字体是设备上下文的属性。 下面列出了一些设备上下文中最常用的属性和访问这些属性的 CNeatDC 函数。 Attribute Default Operation functions 文本颜色 Text Color Black(黑色) \ref CNeatDC::SetTextColor \ref CNeatDC::GetTextColor 背景颜色 Background Color White(黑色) \ref CNeatDC::SetBkColor \ref CNeatDC::GetBkColor 背景模式 Background Mode OPAQUE(覆盖) \ref CNeatDC::SetBkMode \ref CNeatDC::GetBkMode 当前位置 Current Position (0,0) \ref CNeatDC::MoveTo \ref CNeatDC::GetCurrentPosition 31 Attribute Default Operation functions 当前画笔 Current Pen (黑色) (Black) \ref CNeatDC::SelectObject \ref CNeatDC::GetCurrentPen 当前画刷 Current Brush (黑色) (Black) \ref CNeatDC::SelectObject \ref CNeatDC::GetCurrentBrush 当前字体 Current Font (黑色) (Black) \ref CNeatDC::SelectObject \ref CNeatDC::GetCurrentFont 下面我们描述一下怎样创建和使用这些设备上下文。 使用 CNeatClientDC 在窗口客户区进行绘画。 \ref CNeatClientDC 用来在非重绘事件处理函数中对窗口的客户区进行绘制。下面的例子演示了按键时在窗口中随机画线: \code int MyWindow::OnKeyDown(UINT nKeyCode, UINT nRepCnt, UINT nFlags) { int x, y; CNeatRect rect; // 创建一设备上下文 CNeatClientDC dc(this); // 创建一画笔,使用默认属性(宽度为 1,颜色为黑色,实线) CNeatPen pen; // 将画笔选进设备上下文,选进成功后,设备上下文后续的画线将使用这个画笔 dc.SelectObject(&pen); // 获得当前窗口的客户区尺寸,在客户区内随机定位一个,然后画线 GetClientRect( &rect ); x = rand()%rect.Width(); y = rand()%rect.Height(); dc.LineTo(x,y); } \endcode 使用 CNeatPaintDC 在窗口上绘画 如果你定义了一个窗口重绘事件处理函数,则必须在这个处理函数中产生一个 CNeatPaintDC 设备上下文,并且使用它来进行 你需要的绘画动作。 产生这个对象将告诉 NEAT 的窗口体系这个窗口的需要重画的区域已经被重画了, 这样窗口系统就不会重复的 发送重画消息给这个窗口了。重画事件是由于用户和窗口系统的交互造成的,但是它也可以通过调用 CNeatWnd::InvalidateRect 函 数手动产生。 下面的代码演示了如何在窗口正中位置画一个黑边红色的矩形区域,并且会判断这个区域是否位于需要更新的区域范围内以 32 便决定是否需要重画。 \code int MyWindow::OnPaint() { // 创建一设备上下文 CNeatPaintDC dc(this); // 获取窗口大小 CNeatRect rect; GetClientRect( &rect ); // 绘制一矩形框 dc.Rectangle(&rect); } \endcode 绘画工具 画笔 NEAT 使用\ref CNeatPen 来实现画笔的功能,一个画笔对象包含三个属性:画笔的类型,颜色和线宽,默认为:实线,黑色, 1 个像素线宽。如果要使用自定义的画笔,首先要定义一个 CNeatPen 的对象,并设置相关属性;使用时,先要调用\ref CNeatDC::SelectObject 将画笔对象选进设备上下文中,如下面代码所示: \code CNeatPen pen(PT_SOLID,1,COLOR_black); dc.SelectObject(&pen); \endcode 下面给出一个画笔及常用画线(画框)函数的示例代码: \include gdi\pen\src\gdi-pen.cpp \image html gdi-pen-normal.jpg ―画笔 - 普通‖ \image html gdi-pen-width.jpg ―画笔 - 不同线宽‖ \image html gdi-pen-color.jpg ―画笔 - 不同颜色‖ \image html gdi-pen-dbdash.jpg ―画笔 - 双虚线‖ \image html gdi-pen-dash.jpg ―画笔 - 可定制虚线‖ 画刷 NEAT 使用\ref CNeatBrush 来实现画刷的功能,一个画刷对象包含两个属性:画刷的类型和颜色,默认为:纯色(黑色)填 充。 如果要使用自定义的画刷, 首先要定义一个 CNeatBrush 的对象, 并设置相关属性; 使用时, 先要调用\ref CNeatDC::SelectObject 将画笔对象选进设备上下文中,如下面代码所示: \code CNeatBrush brush(COLOR_black); dc.SelectObject(&brush); \endcode 下面给出一个画刷及填充的示例代码: 33 \include gdi\brush\src\gdi-brush.cpp \image html gdi-brush-color.jpg ―画刷 - 纯色填充‖ \image html gdi-brush-CROSS.jpg ―画刷 - 预设类型填充‖ \image html gdi-brush-DIAGCROSS.jpg ―画刷 - 预设类型填充‖ \image html gdi-brush-bmp1.jpg ―画刷 - 位图填充‖ \image html gdi-brush-bmp2.jpg ―画刷 - 位图填充‖ 字体 NEAT 使用\ref CNeatFont 来实现字体的功能,一个字体对象包含属性: 字符集及编码:多字节编码字符集:简体中文(gb2312,gbk),单字节编码字符集:ascii,iso8859-1,ISO8859-15 字体样式:中文默认为宋体,英文默认为(Arial) 字体大小:中文默认为宋体,英文默认为(Arial) 其他属性:下划线,穿透线,粗体,斜体等等 下面给出一个字体使用的示例代码: \include gdi\font\src\gdi-font.cpp 不同大小字体(12,16,24) \image html gdi-font-12.jpg \image html gdi-font-16.jpg \image html gdi-font-24.jpg 不同粗细 \image html gdi-font-bold.jpg \image html gdi-font-demibold.jpg \image html gdi-font-book.jpg \image html gdi-font-subpiexl.jpg 下划线及穿透线斜体 \image html gdi-font-underline.jpg \image html gdi-font-structout.jpg \image html gdi-font-slant.jpg 图标 NEAT 使用\ref CNeatIcon 来实现图标的功能。 位图 NEAT 使用\ref CNeatBitmap 来实现位图的功能。 34 绘制基本图形 下面列出 NEAT 支持的常见的图标图形操作: 函数 功能 备注 \ref CNeatDC::LineTo 画线 当前画笔 \ref CNeatDC::Rectangle 画矩形框 当前画笔 \ref CNeatDC::Circle 画圆 当前画笔 \ref CNeatDC::CircleEx 画圆(支持线类型及线宽) 当前画笔 \ref CNeatDC::Ellipse 画椭圆 当前画笔 \ref CNeatDC::EllipseEx 画椭圆(支持线类型及线宽) 当前画笔 \ref CNeatDC::CircleArc 画弧线 当前画笔 CNeatDC::SetPixel 设置像素颜色 调用参数指定 \ref CNeatDC::FillSolidRect 区域填充 调用参数指定 \ref CNeatDC::FillRect 区域填充,使用当前画刷 当前画刷 \ref CNeatDC::FillCircle 圆填充,使用当前画刷,只支持纯色模式 当前画刷 \ref CNeatDC::FillCircleEx 圆填充,使用当前画刷,支持画刷的各个类型 当前画刷 \ref CNeatDC::FillEllipse 椭圆填充,使用当前画刷,只支持纯色模式 当前画刷 \ref CNeatDC::FillEllipseEx 椭圆填充,使用当前画刷,支持画刷的各个类型 当前画刷 \ref CNeatDC::FillArcEx 弧型填充,使用当前画刷,支持画刷的各个类型 当前画刷 绘制文本 下面列出 NEAT 支持的绘制文本操作: 函数 功能 备注 \ref CNeatDC::TextOut 文本输出 当前画笔 \ref CNeatDC::DrawText 文本格式化输出 当前画笔 \ref CNeatDC::TabbedTextOut 文本输出,支持 TAB 当前画笔 35 文本格式: 格式 备注 格式 备注 DT_TOP 顶部对齐 DT_LEFT 左对齐 DT_CENTER 中间对齐(设置 DT_SINGLELINE 时有效) DT_RIGHT 右对齐 DT_VCENTER 上下居中 DT_BOTTOM 底部对齐 DT_WORDBREAK 自动卷行时,判断单词边界 DT_SINGLELINE 单行 DT_EXPANDTABS TAB 扩展,默认为个空格 DT_TABSTOP 格式参数的高 8 位用来指定 TAB 键宽度 DT_NOCLIP 不进行边界切割 DT_CHARBREAK 当文本输出超过矩形区时按字 符换行输出 DT_CALCRECT 不作实际输出,只计算实际的输 出矩形大小 36 第七章 处理用户输入 PT80 的按键所对应的键值 输入事件相应 事件 消息处理函数 备注 MSG_KEYDOWN \ref CNeatWnd::OnKeyDown 按键按下事件 MSG_KEYUP \ref CNeatWnd::OnKeyUp 按键释放事件 标识 键值 标识 键值 标识 键值 KEY_0 11 KEY_8 9 KEY_OK 59 KEY_1 2 KEY_9 10 KEY_CANCEL 1 KEY_2 3 KEY_UP 103 KEY_POWER 68 KEY_3 4 KEY_DOWN 108 KEY_FUNC 60 KEY_4 5 KEY_LEFT 105 KEY_BACKSPACE 14 KEY_5 6 KEY_RIGHT 106 KEY_TAB 15 KEY_6 7 KEY_ENTER 28 KEY_ALPHA 66 KEY_7 8 KEY_DOT 52 37 第八章 对话框编程基础 使用资源编辑器编辑对话框 几乎每一个 NEAT 程序都会使用对话框与用户进行交互,对话框可能是一个简单的 OK 按钮,也可以是一个复杂的数据输入 表单。NEAT 目前只支持模态对话框方式,\ref CNeatDialog 是对话框的基类,使用模态对话框,在对话框关闭之前,用户不能在 同一应用程序的其他地方工作。 对话框和普通窗口的主要区别在于,对话框几乎始终与资源相关联,这些资源标识对话框元素,并指定它的布局。在 VC 开 发环境下,可以利用 VC 的对话框编辑器(资源编辑器之一)来创建和编辑对话框资源,所以,我们可以快速并且高效地以可视化 的方式生成对话框。 对话框包含许多名为控件的元素,对话框控件包括编辑控件、按钮、列表框、组合框、静态文本(标签)、进度条、滑块等。 控件发送通知消息到它的对话框,以响应键入文本或单击按钮之类的用户活动。 NEAT 已经对这些事件做了很好的封装,使用时,只要重载你关心的事件处理函数就可以了。在对话框创建的时候,要建立 对话框数据成员和这些控件的关联,然后就可以利用这些数据成员进行控件的数据操作了。 模态和非模态对话框编程 模态对话框是最常用的对话框。用户的操作打开一个对话框,用户在对话框中输入数据,然后关闭对话框。下面在当前工程 中增加一个模态对话框的步骤(在 VC 集成开发环境下): 1. 使用对话框编辑器来创建包含不同控件的对话框资源。对话框编辑器更新工程的资源 脚本(RC)文件,以包含新的对话框资源,并且,它使用对应的#define 变量来更新该工程的 resource.h 文件。 2. 创建一个\ref CNeatDialog 的派生类。 3. 在创建的派生类中,添加要进行数据操作的控件数据成员。 4. 在创建的派生类中,添加要处理的控件事件处理函数。 5. 在创建的派生类中,重载\ref CNeatDialog::OnInitDialog 函数,并在此函数里实现控件数据成员和相应控件的关联。 6. 在合适的位置编写代码来激活对话框。这个代码包括对对话框构造函数的调用,接着是对\ref CNeatDialog::DoModal 对 话框类成员函数的调用。只有当用户退出这个对话框窗口时,\ref CNeatDialog::DoModal 函数才返回。 38 通用对话框 对话框 示例 现在,我们将开始一个示例程序。 对话框资源编辑(有关 VC 资源编辑器的使用,已经超出了本帮助文档的范围,但有关 VC 资源编辑器使用介绍的文档 、书籍 很多,请用户去参考相关的使用介绍)。 \image html dialog1-rc.jpg "对话框资源" 创建一个\ref CNeatDialog 的派生类。 \code // CMyDialog // 资源头文件 #include "resource.h" // NEAT 对话框及控件实现相关的头文件 #include // 定义一个对话框 // 创建一个\ref CNeatDialog 的派生类。 class CMyDialog : public CNeatDialog { public: // 构造函数 CMyDialog(UINT dlgid, CNeatWnd* parent); // 对话框创建初始化函数,用于控件的关联和控件数据的初始化。 virtual int OnInitDialog(); // 列表框选择发生改变的事件响应处理函数 virtual int OnLbnSelchange(UINT nID,HWND hwnd); // 确认退出前的事件响应处理函数 virtual int OnOK(); public: wxString m_str; protected: // 列表框控件对象 CNeatListCtrl m_listctrl; }; \endcode 在 OnInitDialog 中实现控件数据成员和相应控件的关联。 \code int CMyDialog::OnInitDialog() { CNeatDialog::OnInitDialog(); 39 // 实现列表控件和列表框对象的关联,IDC_LIST1 为列表框对应的资源 ID m_listctrl.Attach(GetDlgItem(IDC_LIST1)); // 利用列表框对象进行相关的数据操作,这里添加一些字符串 m_listctrl.AddString( _("list string1") ); m_listctrl.AddString( _("list string2") ); m_listctrl.SetCurSel(0); return 0; } \endcode 重写你关心的一些事件处理函数 \code // 列表边框选择发生改变的事件处理函数 int CMyDialog::OnLbnSelchange(UINT nID,HWND hwnd) { if (hwnd==m_listctrl.GetSafeHwnd()) { int sel = m_listctrl.GetCurSel(); } } // 对话确认退出前的事件响应函数 void CMyDialog::OnOK() { int sel = m_listctrl.GetCurSel(); m_str = m_listctrl.GetText(sel); } \endcode 调用这个对话框,看看运行的效果。 \code // 按键事件处理函数 int CMyFrame::OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags) { // 调用对话框 CMyDialog dlg(IDD_DIALOG1,this); if (dlg.DoModal()==IDOK) { NeatMessage( dlg.m_str ); } return 0; } \endcode \image html dialog1-emu.jpg "模拟器下运行效果" 40 对话框资源 在 VC 开发环境下的 NEAT, 可以利用 VC 的资源编辑工具进行资源的编辑, 但没有直接使用它的 rc 文件, 需要使用 neatrg 工 具进行资源的转换,转换后的资源称为资源模板,默认存在 res.cpp 文件中。 资源模板文件 在资源模板文件中,常包含有以下几种资源: 1. 对话框(Dialog)资源,资源数据利用数据结构\ref DLGTEMPLATE 来实现。 2. 位图(Bitmap)资源,资源数据为一个资源 ID 及对应的位图文件。 3. 图标(Icon)资源,资源数据为一个资源 ID 及对应图标文件。 4. 菜单(Menu)资源。 5. 版本信息(Version)资源。 用到了几个将资源数据和资源 ID 建立关联的数据结构: \ref DLGID_TEMPL 用于对话框资源和资源 ID 的关联。 \ref STRID_TEMPL 用于字符串(Bitmap,Icon,cursor 等)资源和资源 ID 的关联。 \ref MENU_TEMPL 用于菜单资源和资源 ID 的关联。 \ref MENUITEM_TEMPL 用于菜单项的关联。 41 第九章 NEAT 控件 控件综述 许多人对控件(或者部件)的概念已经相当熟悉了,控件可以理解为主窗口中的子窗口,这些子窗口的行为和主窗口一样, 既能够接收键盘和鼠标等外部输入,也可以在自己的区域内进行输出,只是它们的所有活动被限制在主窗口中。NEAT 也支持子窗 口,并且可以在子窗口中嵌套建立子窗口。我们将 NEAT 中的所有子窗口均称为控件。 静态框 静态框用来在窗口的特定位置显示文字、数字等信息,还可以用来显示一些静态的图片信息,比如公司徽标、产品商标等等。 就像其名称暗示的那样,静态框的行为不能对用户的输入进行动态的响应,它的存在基本上就是为了展示一些信息,而不会接收任 何键盘或鼠标输入。 静态框风格 静态框的风格由静态框种类和一些标志位组成。我们可将静态框控件按功能划为标准型(只显示文本)、位图型(显示图 标或图片),以及特殊类型组框。下面我们将别介绍上述不同类型的静态框。 标准型 SS_SIMPLE 创建的控件只用来显示单行文本 SS_LEFT 风格创建的静态框可用来显示多行文本并左对齐 SS_CENTER 风格创建的静态框可用来显示多行文本并中对齐 SS_RIGHT 风格创建的静态框可用来显示多行文本并右对齐 SS_LEFTNOWORDWRAP 创建的静态框会扩展文本中的 TAB 符,但不做自动换行处理。 位图型 SS_BITMAP 显示一幅位图 SS_ICON 显示一幅图标 SS_REALSIZEIMAGE 取消缩放操作,并显示在静态框的左上方 SS_CENTERIMAGE 在控件中部显示位图或图标 组框 SS_GROUPBOX 用来包含其他的控件 42 其他静态框类型 SS_WHITERECT 以白色填充静态框矩形 SS_GRAYRECT 以灰色填充静态框矩形 SS_BLACKRECT 以黑色填充静态框矩形 SS_GRAYFRAME 灰色边框 SS_WHITEFRAME 白色边框 SS_BLACKFRAME 黑色边框 编程示例 1. 手动创建: 直接在视图上创建\ref CNeatStatic 的对象,并调用 Create 方法。 2. 利用资源创建: 创建\ref CNeatStatic 的对象并调用 LoadTemplate(CNeatWnd * parent, PCTRLDATA templdata, const wxString & res = NEATAPPNAME) 函数 3. 对话框创建: 首先将 Picture Control 控件拖到模板上 \image html 9.2-static1.JPG 添加图标:设置 Type 熟悉对话框为 Icon,设置 Image 为所选图标的 ID。 添加位图:Type 为 Bitmap。 \image html 9.2-static2.JPG 添加静态框只需将控件拖到模板上即可 \image html 9.2-static3.JPG 部示例代码: \include demo-static.cpp 显示效果截图: \image html demo-static-1.jpg "静态框示例 按键 1" \image html demo-static-2.jpg "静态框示例 按键 2" \image html demo-static-3.jpg "静态框示例 按键 3" \image html demo-static-4.jpg "静态框示例 按键 4" 43 按钮 按钮是除静态框之外使用最为频繁的一种控件。按钮通常用来为用户提供开关选择。NEAT 的按钮可划为普通按钮、复选 框和单选钮等几种类型。用户可以通过键盘或者鼠标来选择或者切换按钮的状态。用户的输入将使按钮产生通知消息,应用程序也 可以向按钮发送消息以改变按钮的状态。 按钮风格 普通按钮 BS_PUSHBUTTON 普通按钮 BS_DEFPUSHBUTTON 默认选中普通按钮 复选框 复选框风格: BS_CHECKBOX 在选中和非选中状态之间切换 BS_AUTOCHECKBOX 控件会自动在选中和非选中状态之间切换 BS_3STATE 能显示第三种状态——复选框内是灰色的,应用程序来操作其状态 BS_AUTO3STATE 能显示第三种状态——复选框内是灰色的,由控件负责状态的自动切换 BS_PUSHLIKE 复选框以普通按钮的形式显示 文本对齐的风格: BS_LEFT 文本左对齐 BS_CENTER 文本水平居中 BS_RIGHT 文本右对齐 BS_TOP 文本上对齐 BS_VCENTER 文本垂直居中 BS_BOTTOM 文本下对齐 单选钮 显示用户的选择情况 BS_RADIOBUTTON 显示用户的选择情况 BS_AUTORADIOBUTTON 自动显示用户的选择情况 BS_PUSHLIKE 单选按钮以普通按钮的形式显示 44 文本对齐的风格: BS_LEFT 文本左对齐 BS_CENTER 文本水平居中 BS_RIGHT 文本右对齐 BS_TOP 文本上对齐 BS_VCENTER 文本垂直居中 BS_BOTTOM 文本下对齐 按钮事件响应 按钮事件 消息处理函数 备注 BN_CLICKED CNeatWnd::OnBnClicked 按钮单击事件的响应函数 BN_PUSHED CNeatWnd::OnBnPushed 按钮按下事件的响应函数 BN_UNPUSH

8,304

社区成员

发帖
与我相关
我的任务
社区描述
游戏开发相关内容讨论专区
社区管理员
  • 游戏开发
  • 呆呆敲代码的小Y
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧