|
作者: ccrun ,如转载请保证本文档的完整性,并注明出处。 O! U K8 i. R# }% Y; K* {3 B
欢迎光临 C++ Builder 研究,http://www.ccrun.com/doc/go.asp?id=592
2 V$ x$ L2 ]: ~; C2 `, z1 c本文是ccrun(老妖)根据偶然间看到的一篇文章略作修改而成。参考资料:0 c) Q `' s" Y2 G0 j
标题:在Delphi中使用IP控件 f/ k! ~( j$ p8 y1 v
作者:西安交通大学9649# 刘明华7 g! R- o$ T& u& N+ K t
在些对原作者的无私奉献表示感谢!如欲转载请保留完整版权信息。) J1 q4 K" f3 s9 [
如果这篇文章侵犯了您的版权,请来信告诉我们:info@ccrun.com
5 g o- p* N' y, @: y8 H% occrun(老妖)(www.ccrun.com) 2003.10.29! r* ^- O( N8 g+ o" q- J& {3 @ Z0 D
' H9 w3 I0 j9 l6 [8 i7 }" p+ ^在网络程序中,我们常常碰到需要用户输入IP地址的情况。然而C++ Builder并没有为我们提供可以用于输入IP串的控件,于是我们只好用TEdit控件(单行文本框)来接受用户输入的IP串。但是,使用TEdit来输入IP串并不是一个好的主意,因为处理起来非常不方便。事实上,在我们的身旁有一个专门用来输入IP串的Windows控件,就象网络链接属性中, Internet 协议 (TCP/IP) 属性中输入IP的控件。IP控件会拒绝非法的IP串(在每个部分只能输入0~255之间的数字);它让你可以轻松地获取控件中的IP串所对应的IP值(32位整数),这省去了IP串和IP值之间相互转换的麻烦;此外,你还能限制IP控件中所能输入的IP的范围。在本文中,我将向大家介绍如何在我们的C++ Builder程序中使用Windows的IP控件。% l* [ @9 G0 U4 V( {
9 B% o% R+ H1 a/ B$ E, C1 F h% v
Windows中有两个非常重要的动态联结库:commctrl.dll和comctl32.dll,它们是Windows的自定义控制库(Windows Common Controls)。自定义控制库中包含了许多常用的Windows控件,如Statusbar,Coolbar,HotKey等;在C++ Builder中,这些控件大多数都已被包装成可视化控件了。在Microsoft推出Internet Explorer 3之后,自定义控制库中新增了一些控件,其中就包括Windows的IP控件(IP Address edit control)。
6 B3 c: ]$ I# ?: i# J [) _8 ]/ A C( {& r
●初始化Windows自定义控制库●
& [9 c3 d" B0 N$ R; ^% W0 N/ JWindows提供了两个API函数,InitCommonControls和InitCommonControlsEx,用来初始化自定义控制库。从名字我们不难看出这两个API函数的关系:后者是前者的增强。如果你希望在程序中使用IP控件,你必须用InitCommonControlsEx来完成对自定义控制库以及类的初始化。函数InitCommonControlsEx的原型如下:6 s* Q2 s4 T O) B3 ?& Y( U
4 n; g2 }* e7 t
typedef struct tagINITCOMMONCONTROLSEX {
1 ~- R) C, o0 U- V7 q DWORD dwSize; // size of this structure2 G( O' J/ m m( D) w
DWORD dwICC; // flags indicating which classes to be initialized
' g) m/ @6 H9 _2 D, U+ L3 J: l} INITCOMMONCONTROLSEX, *LPINITCOMMONCONTROLSEX;
4 \" b' R! P5 \5 W! K0 x9 L3 f# [& N- V
WINCOMMCTRLAPI BOOL WINAPI InitCommonControlsEx(LPINITCOMMONCONTROLSEX);4 G- y: f& s! S9 I
3 a @; k, T3 YIP控件属于ICC_INTERNET_CLASSES类别的控件,如果要使用该控件,你应该在程序中包含如下的初始化代码:6 \) w* d0 B6 P8 o& L4 c
+ k `. I8 c; b+ pTInitCommonControlsEx ICC;
1 k5 P& O& s' u5 aICC.dwSize = sizeof(TInitCommonControlsEx);5 w% q% k/ }% N& h7 p: G
ICC.dwICC = ICC_INTERNET_CLASSES;
0 E& Z( f, w' d, U, L4 Eif(!InitCommonControlsEx(&ICC))3 H6 U0 z& R# Y! j
return; //初始化组件失败
_# k5 l& R3 {, x9 v3 s4 m# t& v* V7 S6 u# \
●创建IP控件●
: E! r2 v7 \& K5 e5 V" k5 {& eWindows API函数CreateWindow或者CreateWindowEx都可以用来创建一个IP控件实例。IP控件的窗口类名为"SysIPAddress32",C++ Builder的commctrl.pas单元为其定义了一个符号常量WC_IPADDRESS。下面这条语句将在Form1上创建一个IP控件。! e3 _ Q5 P5 e1 Q2 q
, A' p6 M' J+ F5 q% fHWND hIpEdit = CreateWindow(WC_IPADDRESS,NULL,WS_CHILD|WS_VISIBLE,10,10,135,47,Handle,0,HInstance,NULL);( @5 P0 g) T5 _: p9 j* M7 j7 G
3 _. W( }% b9 E, p, _! N, }6 Q% L●使用IP控件●6 M& F' \0 S/ w7 X& o
在程序中,我们通过向IP控件发送消息来与它通讯。IP控件可以响应的消息有以下6个,这些消息及它们的含义见下表:) Z& J1 U# [3 l1 D' x
9 N$ X- Q7 e3 [8 |; S) z0 S3 A消息常量 消息值 作用 参数及返回值
$ Y$ a% g; m. V& `4 BIPM_CLEARADDRESS WM_USER+100 清除IP控件中的IP串 参数无- y3 ~7 h* t# v5 D+ P0 I% g+ v
IPM_SETADDRESS WM_USER+101 设置IP控件的IP串 Lparam为32位的IP值
8 j$ f# K9 a) S8 @) G1 aIPM_GETADDRESS WM_USER+102 获取IP控件中的IP串所对应的IP值(32位整数) Lparam为一个指向Integer变量的指针。返回值等于IP控件中非控的字段数目;获取到的IP值存放在lparam 所指向的Integer变量中。
/ [2 }7 r6 D* _ l& CIPM_SETRANGE WM_USER+103 设置IP控件4个部分的其中一个的IP取值范围 Wparam指明要设置取值范围的部分;lparam的低16位字为该字段的范围:高字节为上限,低字节为下限。% z/ f$ S3 y- M$ `9 K, _
IPM_SETFOCUS WM_USER+104 设输入焦点 Wparam指明哪个部分获取焦点
- a: i4 J3 J0 N `IPM_ISBLANK WM_USER+105 IP串是否为空 参数无。返回值:若为空,返回非0;不为空,返回0; P8 _. S# g2 }" K( ~( ^" e, |
1 f( f; f; H% v+ K
★㈠清空IP串(IPM_CLEARADDRESS)★1 O0 F' N6 C8 x5 Y/ q$ l2 |5 X
SendMessage(hIpEdit,IPM_CLEARADDRESS,0,0);
4 D$ N% _8 F1 B9 S0 n0 M
7 s$ g1 O9 H$ }/ J* I6 ^$ r3 {: |★㈡设置IP串(IPM_SETADDRESS)★! T# z- w) e# y# p! ? b% b
int nIP;
^9 z* g# i4 C; S, T4 ^7 U1 pnIP=MAKEIPADDRESS(192,168,0,1);
% ~3 J0 R3 y6 hSendMessage(hIpEdit,IPM_SETADDRESS,0,nIP);
1 D: g: @! X8 }" O, q N% r8 z6 W" t
. Z! e: w, p. i; o此例将IP控件的内容设为"192.168.0.1",其中MAKEIPADDRESS是一个Win32宏,定义在commctrl.h头文件中,它用来合成一个32位的IP值:$ g* _2 X* n# f
#define MAKEIPADDRESS(b1,b2,b3,b4)! a7 y* j: N. ]5 W; ~9 W
((LPARAM)(((DWORD)(b1)<<24)+((DWORD)(b2)<<16)+((DWORD)(b3)<<8)+((DWORD)(b4))))9 r# u0 k/ d9 L) s
& N3 z2 z8 L# N2 p& a★㈢获取IP值(IPM_GETADDRESS)★
3 `) |- D+ ^1 p) \int nIP;
6 y% B/ v( B' R2 ESendMessage(hIpEdit,IPM_GETADDRESS,0,int(&nIP));, s }3 B/ M o- O
//nIP++;! f* y% q1 B/ N/ d& t
//SendMessage(hIpEdit,IPM_SETADDRESS,0,nIP); //将IP加1再赋给IP控件。1 u' s; ^ |; E H2 F) ^ V h
- V+ g; F3 D' q. `( i若想要获取IP控件中IP串所对应的IP值,你应该向IP控件发送IPM_GETADDRESS消息,并且需要把一个32位整数的地址作为SendMessage的最后一个参数。
( w# {. {/ h0 C: j0 ]
- C! P" R. G" U+ G/ _8 T5 }★㈣设置取值范围(IPM_SETRANGE)★
. Z4 ^6 ?8 Q2 f8 {; DSendMessage (hIpEdit, IPM_SETRANGE, 0, 200<<8|100);
0 h$ @; }. |6 [) @- s B. ]此语句将IP控件的第一部分的范围限制为100~200。在IPM_SETRANGE消息中,Wparam指明要设置的字段, 而lparam的低16位字为该字段的范围:高字节为上限,低字节为下限。
; P. E {5 @0 M, o7 m ] ^% U, Y4 A) D" \
★㈤设置输入焦点(IPM_SETFOCUS)★
) U$ A' Y) w# U ^# [SendMessage(hIpEdit,IPM_SETFOCUS,3,0);& a( o$ N$ b/ }+ y" a
//将输入焦点设在IP控件的第四部分。
& q, U6 }! }/ x# |5 u3 s w0 @
㈥判断IP串是否为空(IPM_ISBLANK)
& x6 z$ b3 V3 b% R- Aif(!SendMessage(hIpEdit,IPM_ISBLANK,0,0)) 8 ~4 h, [* R' k1 i
{
! f# l- ?% ^6 a; F# U, X //IP控件中的IP串为空
& E- E/ ~ ^( \4 g}( n3 o5 \7 A. S$ @$ w
else& ^1 S( T5 [- X4 v
{8 f) x4 V: S: a" _2 L
//IP控件中的IP串至少有一部分不是空的2 V- u+ I: W5 c; p0 y. R" s
}; E6 @. D \5 i; j- g4 E
9 M# U' m( W' G●IP控件的通知消息●
# I3 k$ |$ ?0 B- f% N9 X3 s当IP串被改动后或者输入焦点发生了转移,IP控件就会向它的父窗口发送通知消息IPN_FIELDCHANGED。在大多数情况下,我们都可以忽略此通知消息。以下是处理通知消息IPN_FIELDCHANGED的一个示例:
# u+ d' Q k5 G# L0 J6 e
9 z/ G+ B& L* `. k2 Evoid __fastcall TForm1::WndProc(TMessage &Msg)$ G* u0 P5 C- g& i W$ ]1 w8 h- e: Z
{2 X" f2 d% B6 U! n/ e
LPNMHDR p=(LPNMHDR)Msg.LParam;
4 p5 d5 E9 d" w( R if(Msg.Msg==WM_NOTIFY)
+ t# L7 x3 ~. | {7 E. V, g. `! T' L$ h
if(p->code==IPN_FIELDCHANGED)
; n3 B* u& U0 c/ W2 Z, u {( }4 z# B4 \, F* ?) \% ^) Y l
//处理IP控件的IPN_FIELDCHANGED通知消息
5 J9 [. y# w w$ R }$ z4 K( D0 g3 o
}4 P# ?* z8 `' Z0 Y+ |3 r0 h/ M% d
TForm::WndProc(Msg);$ |. }* n- B" w. p8 F( p$ o" t
} |
|