﻿<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/" xmlns:wfw="http://wellformedweb.org/CommentAPI/" xmlns:slash="http://purl.org/rss/1.0/modules/slash/"><channel><title>C++博客-被窝里的流年-随笔分类-MFC</title><link>http://www.cppblog.com/cs569652894/category/16566.html</link><description>在CR学习的点点滴滴，希望能够给那些还在举步维艰的人给予一点点帮助！一起共同进步！加油！</description><language>zh-cn</language><lastBuildDate>Fri, 08 Jul 2011 23:26:31 GMT</lastBuildDate><pubDate>Fri, 08 Jul 2011 23:26:31 GMT</pubDate><ttl>60</ttl><item><title>TCP,UDP完整数据包校验和通用计算</title><link>http://www.cppblog.com/cs569652894/archive/2011/07/06/150264.html</link><dc:creator>CrackerCS</dc:creator><author>CrackerCS</author><pubDate>Tue, 05 Jul 2011 17:42:00 GMT</pubDate><guid>http://www.cppblog.com/cs569652894/archive/2011/07/06/150264.html</guid><wfw:comment>http://www.cppblog.com/cs569652894/comments/150264.html</wfw:comment><comments>http://www.cppblog.com/cs569652894/archive/2011/07/06/150264.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/cs569652894/comments/commentRss/150264.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/cs569652894/services/trackbacks/150264.html</trackback:ping><description><![CDATA[TCP,UDP完整数据包校验和通用计算
<div class="article" id="blog_article"><table cellspacing="0" cellpadding="0"><tbody><tr><td class="l_text"><div> <wbr> <wbr> <wbr> ICMP，IP,UDP,TCP报头部分都有checksum（检验和）字段。ICMP和IP报头校验和的计算都很简单，使用RFC1071中给出的方法即可完成（如下）。</wbr></wbr></wbr></div><div> <wbr></wbr></div><div><strong><font color="#0000f0">//计算校验和<br />USHORT checksum(USHORT *buffer,int size)<br />{<br /> <wbr>unsigned long cksum=0;<br /> <wbr>while(size&gt;1)<br /> <wbr>{<br /> <wbr> <wbr>cksum+=*buffer++;<br /> <wbr> <wbr>size-=sizeof(USHORT);<br /> <wbr>}<br /> <wbr>if(size)<br /> <wbr>{<br /> <wbr> <wbr>cksum+=*(UCHAR *)buffer;<br /> <wbr>}<br /> <wbr>//将32位数转换成16<br /> <wbr>while (cksum&gt;&gt;16)<br /> <wbr> <wbr>cksum=(cksum&gt;&gt;16)+(cksum &amp; 0xffff);<br /> <wbr>return (USHORT) (~cksum);<br />}<br /></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></font></strong></div><div> <wbr> <wbr> <wbr> UDP/TCP报头中的校验和的计算比较复杂的，要用到 UDP/TCP伪首部：先要填充伪首部各个字段，然后再将UDP/TCP报头以后（包括报头）的数据附加到伪首部的后面，再对位首部使用上述校验和计算，所得到的值才是UDP/TCP报头部分的校验和。</wbr></wbr></wbr></div><div> <wbr></wbr></div><div>位首部可以用如下的结构体表示：</div><div><strong><font color="#0000f0">typedef struct{<br /> <wbr>ULONG <wbr> sourceip; <wbr> <wbr> <wbr> //源IP地址<br /> <wbr>ULONG <wbr> destip; <wbr> <wbr> <wbr> <wbr> <wbr> //目的IP地址<br /> <wbr>BYTE mbz; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //置空(0)<br /> <wbr>BYTE ptcl; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //协议类型<br /> <wbr>USHORT plen; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //TCP/UDP数据包的长度(即从TCP/UDP报头算起到数据包结束的长度 单位:字节)<br />}Psd_Header;</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></font></strong></div><div> <wbr></wbr></div><div> <wbr></wbr></div><div>这个过程是一个很繁琐的过程，计算过几次后再也忍受不了做这样重复的工作，于是写了一个通用的计算函数。这个函数使用起来我感觉非常方便：先封装好你的数据包（完整的，包括以太头），然后将数据包的首地址作为参数，调用该函数即可。函数将帮你完成IP报头以及UDP/TCP报头部分校验和的计算。</div><div> <wbr></wbr></div><div><strong><font color="#0000f0">//-------------------------------------------------------------------------<br />// PacketCheckSum<br />// 计算数据包的校验和<br />// 参数:packet-待处理数据(将封装好的数据包的指针)<br />//-------------------------------------------------------------------------<br />void PacketCheckSum(unsigned char packet[])<br />{<br /> <wbr>Dlc_Header *pdlc_header=NULL; //以太头指针<br /> <wbr>Ip_Header <wbr> *pip_header=NULL; <wbr> //IP头指针<br /> <wbr>unsigned short attachsize=0; //传输层协议头以及附加数据的总长度</wbr></wbr></wbr></wbr></wbr></font></strong></div><div><strong><font color="#0000f0"> <wbr>pdlc_header=(Dlc_Header *)packet;</wbr></font></strong></div><div><strong><font color="#0000f0"> <wbr>//判断ethertype,如果不是IP包则不予处理<br /> <wbr>if(ntohs(pdlc_header-&gt;ethertype)!=0x0800) return;</wbr></wbr></font></strong></div><div><strong><font color="#0000f0"> <wbr>pip_header=(Ip_Header <wbr> *)(packet+14);<br /> <wbr>//TCP包<br /> <wbr>if(0x06==pip_header-&gt;proto)<br /> <wbr>{<br /> <wbr> <wbr><br /> <wbr> <wbr>Tcp_Header *ptcp_header=NULL; //TCP头指针<br /> <wbr> <wbr>Tcp_Psd_Header *ptcp_psd_header=NULL;<br /> <wbr> <wbr><br /> <wbr> <wbr>ptcp_header=(Tcp_Header *)(packet+14+((pip_header-&gt;ver_len)&amp;15)*4);</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></font></strong></div><div><strong><font color="#0000f0"> <wbr> <wbr>attachsize=ntohs(pip_header-&gt;total_len)-((pip_header-&gt;ver_len)&amp;15)*4;<br /> <wbr> <wbr>ptcp_psd_header=(Tcp_Psd_Header *)malloc(attachsize+sizeof(Tcp_Psd_Header));<br /> <wbr> <wbr>if(!ptcp_psd_header) return;<br /> <wbr> <wbr>memset(ptcp_psd_header,0,attachsize+sizeof(Tcp_Psd_Header));</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></font></strong></div><div><strong><font color="#0000f0"> <wbr> <wbr>//填充伪TCP头<br /> <wbr> <wbr>ptcp_psd_header-&gt;destip=pip_header-&gt;destIP;<br /> <wbr> <wbr>ptcp_psd_header-&gt;sourceip=pip_header-&gt;sourceIP;<br /> <wbr> <wbr>ptcp_psd_header-&gt;mbz=0;<br /> <wbr> <wbr>ptcp_psd_header-&gt;ptcl=0x06;<br /> <wbr> <wbr>ptcp_psd_header-&gt;tcpl=htons(attachsize);</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></font></strong></div><div><strong><font color="#0000f0"> <wbr> <wbr>//计算TCP校验和<br /> <wbr> <wbr>ptcp_header-&gt;chksum=0;<br /> <wbr> <wbr>memcpy((unsigned char *)ptcp_psd_header+sizeof(Tcp_Psd_Header),<br /> <wbr> <wbr> <wbr>(unsigned char *)ptcp_header,attachsize);<br /> <wbr> <wbr>ptcp_header-&gt;chksum=checksum((unsigned short *)ptcp_psd_header,<br /> <wbr> <wbr> <wbr>attachsize+sizeof(Tcp_Psd_Header));<br /> <wbr> <wbr><br /> <wbr> <wbr>//计算ip头的校验和<br /> <wbr> <wbr>pip_header-&gt;checksum=0;<br /> <wbr> <wbr>pip_header-&gt;checksum=checksum((unsigned short *)pip_header,20);<br /> <wbr> <wbr>return;<br /> <wbr>}<br /> <wbr><br /> <wbr>//UDP包<br /> <wbr>if(0x11==pip_header-&gt;proto)<br /> <wbr>{<br /> <wbr> <wbr>Udp_Header *pudp_header=NULL; //UDP头指针<br /> <wbr> <wbr>Udp_Psd_Header *pudp_psd_header=NULL;</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></font></strong></div><div><strong><font color="#0000f0"> <wbr> <wbr>pudp_header=(Udp_Header *)(packet+14+((pip_header-&gt;ver_len)&amp;15)*4);</wbr></wbr></font></strong></div><div><strong><font color="#0000f0"> <wbr> <wbr>attachsize=ntohs(pip_header-&gt;total_len)-((pip_header-&gt;ver_len)&amp;15)*4;<br /> <wbr> <wbr>pudp_psd_header=(Udp_Psd_Header *)malloc(attachsize+sizeof(Udp_Psd_Header));<br /> <wbr> <wbr>if(!pudp_psd_header) return;<br /> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> memset(pudp_psd_header,0,attachsize+sizeof(Udp_Psd_Header));</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></font></strong></div><div><strong><font color="#0000f0"> <wbr> <wbr>//填充伪UDP头<br /> <wbr> <wbr>pudp_psd_header-&gt;destip=pip_header-&gt;destIP;<br /> <wbr> <wbr>pudp_psd_header-&gt;sourceip=pip_header-&gt;sourceIP;<br /> <wbr> <wbr>pudp_psd_header-&gt;mbz=0;<br /> <wbr> <wbr>pudp_psd_header-&gt;ptcl=0x11;<br /> <wbr> <wbr>pudp_psd_header-&gt;udpl=htons(attachsize);<br /> <wbr> <wbr><br /> <wbr> <wbr>//计算UDP校验和<br /> <wbr> <wbr>pudp_header-&gt;chksum=0;<br /> <wbr> <wbr>memcpy((unsigned char *)pudp_psd_header+sizeof(Udp_Psd_Header),<br /> <wbr> <wbr> <wbr>(unsigned char *)pudp_header,attachsize);<br /> <wbr> <wbr>pudp_header-&gt;chksum=checksum((unsigned short *)pudp_psd_header,<br /> <wbr> <wbr> <wbr>attachsize+sizeof(Udp_Psd_Header));<br /> <wbr> <wbr> <wbr> <wbr><br /> <wbr> <wbr>//计算ip头的校验和<br /> <wbr> <wbr>pip_header-&gt;checksum=0;<br /> <wbr> <wbr>pip_header-&gt;checksum=checksum((unsigned short *)pip_header,20); <wbr> <wbr><br /> <wbr> <wbr>return;<br /> <wbr>}<br /> <wbr>return;<br />}</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></font></strong></div><div> <wbr></wbr></div><div><strong><font color="#000000">需要几个头文件，以及库：</font></strong></div><div> <wbr></wbr></div><div><strong><font color="#f00000">#include &lt;winsock2.h&gt;<br />#include &lt;windows.h&gt;<br />#include "packet.h"<br />#pragma comment(lib,"ws2_32.lib")</font></strong></div><div> <wbr></wbr></div><div> <wbr></wbr></div><div> <wbr></wbr></div><div>最后附上我使用的数据包的结构体（比较多）：</div><div> <wbr></wbr></div><div> <wbr></wbr></div><div>//数据包结构体<br />#pragma pack(1) <wbr><br /><br />typedef struct {<br /> <wbr> <wbr> BYTE <wbr> desmac[6]; <wbr> <wbr> <wbr> <wbr> <wbr> //目的MAC地址<br /> <wbr> <wbr> BYTE <wbr> srcmac[6]; <wbr> <wbr> <wbr> <wbr> <wbr> //源MAC地址<br /> <wbr> <wbr> USHORT <wbr> ethertype; <wbr> <wbr> <wbr> //帧类型<br />}Dlc_Header;</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></div><div><br />typedef struct {<br /> <wbr> <wbr> USHORT hw_type; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //硬件类型Ethernet:0x1<br /> <wbr> <wbr> USHORT prot_type; <wbr> <wbr> <wbr> <wbr> //上层协议类型IP:0x0800<br /> <wbr> <wbr> BYTE hw_addr_len; <wbr> <wbr> <wbr> <wbr> //硬件地址长度:6<br /> <wbr> <wbr> BYTE prot_addr_len; <wbr> <wbr> //协议地址(IP地址)的长度:4<br /> <wbr> <wbr> USHORT flag; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //1表示请求,2表示应答<br /> <wbr> <wbr> BYTE send_hw_addr[6]; //源MAC地址<br /> <wbr> <wbr> UINT send_prot_addr; <wbr> //源IP地址<br /> <wbr> <wbr> BYTE targ_hw_addr[6]; //目的MAC地址<br /> <wbr> <wbr> UINT targ_prot_addr; <wbr> //目的IP地址<br /> <wbr> <wbr> BYTE padding[18]; <wbr> <wbr> <wbr> <wbr> //填充数据 <wbr> <wbr><br />}Arp_Frame;</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></div><div><br />typedef struct {<br /> <wbr>Dlc_Header dlcheader;//DLC头<br /> <wbr>Arp_Frame arpframe; <wbr> //ARP帧<br />}ARP_Packet;</wbr></wbr></wbr></div><div><br />typedef struct {<br /> <wbr>BYTE <wbr> ver_len; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //IP包头部长度,单位：4字节<br /> <wbr>BYTE <wbr> tos; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //服务类型TOS<br /> <wbr>USHORT total_len; <wbr> <wbr> <wbr> //IP包总长度 <wbr><br /> <wbr>USHORT ident; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //标识<br /> <wbr>USHORT frag_and_flags; <wbr> //标志位<br /> <wbr>BYTE ttl; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //生存时间<br /> <wbr>BYTE proto; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //协议<br /> <wbr>USHORT checksum; <wbr> <wbr> <wbr> //IP首部校验和<br /> <wbr>UINT <wbr> sourceIP; <wbr> //源IP地址(32位)<br /> <wbr>UINT <wbr> destIP; <wbr> <wbr> <wbr> //目的IP地址(32位)<br />}Ip_Header;</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></div><div><br />typedef struct {<br /> <wbr>USHORT srcport; <wbr> <wbr> // 源端口<br /> <wbr>USHORT dstport; <wbr> <wbr> // 目的端口<br /> <wbr>UINT seqnum; <wbr> <wbr> <wbr> <wbr> <wbr> // 顺序号<br /> <wbr>UINT acknum; <wbr> <wbr> <wbr> <wbr> <wbr> // 确认号<br /> <wbr>BYTE dataoff; <wbr> <wbr> <wbr> <wbr> // TCP头长<br /> <wbr>BYTE flags; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> // 标志（URG、ACK等）<br /> <wbr>USHORT window; <wbr> <wbr> <wbr> // 窗口大小<br /> <wbr>USHORT chksum; <wbr> <wbr> <wbr> // 校验和<br /> <wbr>USHORT urgptr; <wbr> <wbr> <wbr> // 紧急指针<br />}Tcp_Header;</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></div><div>//TCP伪首部 用于进行TCP校验和的计算,保证TCP效验的有效性<br />typedef struct{<br /> <wbr>ULONG <wbr> sourceip; <wbr> <wbr> <wbr> //源IP地址<br /> <wbr>ULONG <wbr> destip; <wbr> <wbr> <wbr> <wbr> <wbr> //目的IP地址<br /> <wbr>BYTE mbz; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //置空(0)<br /> <wbr>BYTE ptcl; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //协议类型(IPPROTO_TCP)<br /> <wbr>USHORT tcpl; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //TCP包的总长度(单位:字节)<br />}Tcp_Psd_Header;</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></div><div><br />typedef struct <wbr> { <wbr><br /> <wbr>USHORT srcport; <wbr> <wbr> <wbr> <wbr> // 源端口<br /> <wbr>USHORT dstport; <wbr> <wbr> <wbr> <wbr> // 目的端口<br /> <wbr>USHORT total_len; <wbr> <wbr> // 包括UDP报头及UDP数据的长度(单位:字节)<br /> <wbr>USHORT chksum; <wbr> <wbr> <wbr> <wbr> <wbr> // 校验和<br />}Udp_Header;</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></div><div><br />typedef struct tsd_hdr <wbr><br />{ <wbr><br /> <wbr>ULONG <wbr> sourceip; <wbr> <wbr> <wbr> //源IP地址<br /> <wbr>ULONG <wbr> destip; <wbr> <wbr> <wbr> <wbr> <wbr> //目的IP地址<br /> <wbr>BYTE <wbr> mbz; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //置空(0)<br /> <wbr>BYTE <wbr> ptcl; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //协议类型(IPPROTO_UDP)<br /> <wbr>USHORT udpl; <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> <wbr> //UDP包总长度(单位:字节) <wbr><br />}Udp_Psd_Header;</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></div><div><br />typedef struct{<br /> <wbr>BYTE i_type; <wbr> <wbr> <wbr> <wbr> //类型 类型是关键:0-&gt;回送应答(Ping应答) 8-&gt;回送请求(Ping请求)<br /> <wbr>BYTE i_code; <wbr> <wbr> <wbr> <wbr> //代码 这个与类型有关 当类型为0或8时这里都是0<br /> <wbr>USHORT i_cksum; <wbr> //ICMP包校验和<br /> <wbr>USHORT i_id; <wbr> <wbr> <wbr> <wbr> //识别号(一般用进程ID作为标识号)<br /> <wbr>USHORT i_seq; <wbr> <wbr> <wbr> //报文序列号(一般设置为0)<br /> <wbr>//UINT timestamp; <wbr> //时间戳<br /> <wbr>BYTE padding[32];//填充数据<br />}Icmp_Header;<br /><br />typedef struct<br />{<br /> <wbr>Dlc_Header dlc_header; <wbr> //以太帧<br /> <wbr>Ip_Header <wbr> ip_header; <wbr> <wbr> //IP头<br /> <wbr>Icmp_Header icmp_header;//ICMP帧<br />}Icmp_Packet;</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></div><div><br />typedef struct<br />{<br /> <wbr>unsigned char flag; <wbr> <wbr> <wbr> <wbr> //攻击数据包类型1-arp,2-tcp,3-udp<br /> <wbr>unsigned int srcip; <wbr> <wbr> <wbr> <wbr> //攻击者IP<br /> <wbr>unsigned char code[33]; //攻击特征码<br />}Attack_Infor;<br />#pragma pack() <wbr><br /></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></div></td></tr></tbody></table></div><img src ="http://www.cppblog.com/cs569652894/aggbug/150264.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/cs569652894/" target="_blank">CrackerCS</a> 2011-07-06 01:42 <a href="http://www.cppblog.com/cs569652894/archive/2011/07/06/150264.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>mfc控件总览</title><link>http://www.cppblog.com/cs569652894/archive/2011/05/06/145785.html</link><dc:creator>CrackerCS</dc:creator><author>CrackerCS</author><pubDate>Thu, 05 May 2011 16:45:00 GMT</pubDate><guid>http://www.cppblog.com/cs569652894/archive/2011/05/06/145785.html</guid><wfw:comment>http://www.cppblog.com/cs569652894/comments/145785.html</wfw:comment><comments>http://www.cppblog.com/cs569652894/archive/2011/05/06/145785.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/cs569652894/comments/commentRss/145785.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/cs569652894/services/trackbacks/145785.html</trackback:ping><description><![CDATA[
		<p>
				<br />1。mfc单文档结构<br />2。添加 左击鼠标，按下键盘事件<br />3。消息：窗口消息，控件的通知消息，命令消息<br />第5章对话框和常用控件 <br />5.1 对话框的使用 <br />5.2 控件的创建和使用方法 <br />5.3 常用控件 <br />5.4 通用对话框和消息对话框 <br />5.1对话框的使用<br />5.1.1模式和无模式对话框 <br />Ø    模式对话框 <br />u      指当对话框被弹出，用户必须在对话框中作出相应的操作，在退出对话框之前，对话框所在的应用程序不能继续执行。<br />u      无模式对话框 <br />u      指当对话框被弹出后，一直保留在屏幕上，可继续在对话框所在的应用程序中进行其它操作；需要使用对话框时，单击对话框所在的区域即可激活。<br />在创建和退出对话框窗口时的方式不同:<br />u      创建时，模式对话框由系统自动分配内存空间，对话框退出时，对话框对象自动删除。无模式对话框需要来指定内存，退出时还需删除对话框对象。<br />u      退出时，两种对话框所使用的终止函数不一样。模式对话框通过调用CDialog:: EndDialog来终止，无模式对话框则是调用CWnd::DestroyWindow来终止。<br />u      函数CDiaolog::OnOK和CDiaolog::OnCancel是调用EndDialog的，因此无模式对话框必须用DestroyWindow来重载OnOK 和OnCancel两个函数。<br />u      需要正确删除表示对话框的对象。对模式对话框，在创建函数返回后即可删除对象。无模式对话框不是同步的，在创建函数调用后立即返回，因而不知道何时删除对象，但可以通过重载CWnd::PostNcDestroy函数并执行清除操作。  <br />5.1对话框的使用<br />5.1.2使用对话框编辑器 <br />Ø     打开对话框编辑器 <br />     将项目工作区窗口切换到ResourceView页面，双击Dialog目录下任意一个对话框ID。或者，选择“Insert”à“Resource”菜单命令(或按快捷键Ctrl+R)，选择Dialog项，单击New。 <br />5.1对话框的使用<br />在对话框中添加和编辑控件 <br />     (1)控件添加的方法 <br />     在对话框中添加一个控件的方法有下列几种：<br />u       在控件工具栏中单击某控件，鼠标箭头在对话框内变成“十”字形状；在对话框指定位置单击鼠标左键，再拖动选择框可改变控件的大小和位置。<br />u       在控件工具栏中单击某控件，鼠标箭头对话框内变成“十”字形状；在指定位置处单击鼠标左键不放，拖动鼠标至满意位置，释放鼠标键。<br />u       用鼠标左键点中控件工具栏中的某控件，并按住鼠标键不放；在移动鼠标到对话框的指定位置的过程中，会看到一个虚线框，下面带有该控件的标记；释放鼠标键。 <br />5.1对话框的使用<br />(2)单个控件的选取<br />Ø鼠标选取<br />                保证选择按钮(   )是被选中的，移动鼠标指针至指定的控件上，单击鼠标左键。<br />Ø助记符选取<br />                在相应的控件上右击。选择“Properties”菜单项，对话框中的“Caption”框是用来输入控件的标题，在标题中的某个字母之前键入“&amp;”符号，则该字母就是此控件的助记符。<br />ØTab键选取<br />                利用Tab键，可在对话框内的控件中进行选择。每按一次Tab键依次选取对话框中的下一个控件，若按住Shift键，再单击Tab键则选取上一个控件。 <br />(3)多个控件的选取 <br />u在对话框内按住鼠标左键不放，拖出一个大的虚框，释放鼠标，则被该虚框所包围的控件都将被选取。<br />u先按住Shift键不放，用鼠标选取控件，直到所需要的多个控件选取之后再释放Shift键。若在选取时，对已选取的控件再选取一下，则该控件取消选取。<br />u一旦单个控件被选取后，其四周由选择框包围着，选择框上还有几个(通常是八个)蓝色实心小方块，拖动它可改变控件的大小，如图。<br />u多个控件被选取后，其中只有一个控件的选择框有几个蓝色实心小方块，这个控件称为主要控件，而其他控件的选择框的小方块是空心的。如图。<br />5.1对话框的使用<br />(4)控件的移动、删除和复制<br />     当单个控件或多个控件被选取后，按方向键或用鼠标拖动控件的选择框可移动控件。若在鼠标拖动过程中还按住Ctrl键则复制控件。若按Del键可将选取的控件删除。 <br />(5)设定控件的Tab键次序<br />     一是改变用Tab键选择控件的次序，二是当两个或两个以上的控件构成一组时，需要它们的Tab键次序连续。<br />     对话框内控件的Tab键次序设定步骤如下：<br />u      首先，选择“Layout”à“Tab Order”命令，或按Ctrl+D，此时每个控件的左上方都有一个数字，表明了当前Tab键次序，如图。<br />u      其次，按新的次序依次用鼠标单击各个控件，新的Tab键即可生成。<br />u      最后，单击对话框或按Enter键结束Tab Order 方式。<br />5.1对话框的使用<br />对话框内的控件的布局 <br />5.1对话框的使用<br />u      大多布置控件的命令使用前，需要选取多个控件，且“主要控件”起到了关键作用。多个控件的布置过程中，常需要重新设置“主要控件”。设置的方法是按住Ctrl或Shift键，用鼠标单击所要指定的控件。<br />u      系统提供了网格、标尺等辅助工具。在图5.5的控件布局工具栏的最后两个按钮分别用来网格和标尺的切换。一旦网格显示，添加或移动控件时都将自动定位在网格线上。“Layout”菜单下的“Guide Settings”命令提供了设置网格单元大小的功能，如图。<br />Ø     测试对话框<br />5.1对话框的使用<br />5.1.3对话框的编程 <br />Ø     添加对话框资源 <br />     选择“Insert”à“Resource”菜单命令(或按Ctrl+R)，若单击Dialog项左边的“+”号，将展开对话框资源的不同类型选项，如图。  <br />5.1对话框的使用<br />Ø    改变对话框的属性 <br />     在对话框模板的非控件的区域内左击，用下列方式之一打开属性对话框：<br />u      选择“View”菜单à“Properties”命令或按Alt+Enter健。<br />u      用鼠标右击对话框模板，选取Properties命令。<br />     如图，General是对话框属性的默认显示页面，其中列有对话框的标识符名称框、标题框等，具体说明如表。<br />5.1对话框的使用<br />Ø    向对话框内添加控件 <br />Ø    添加一个类 ：注意：以字母c开头<br />     在对话框模板的非控件的区域内双击，如图的对话框。 <br />     单击[OK]，弹出如图的对话框。可以为对话框资源定义一个新类名c123。 <br />5.1对话框的使用<br />Ø    消息映射 首先添加“按钮”<br />     单击[OK]按钮，出现“MFC ClassWizard”对话框，如图。 <br />5.1对话框的使用<br />     选定IDC_BUTTON1的BN_CLICKED，双击左键或单击[Add Function]按钮，如图。<br />     对话框中，可以为添加的成员函数名重新定义，单击[OK]。选定成员函数OnButton1项，双击左键或单击[Edit Code]按钮，在此成员函数添加代码。 <br />u      不同资源对象(控件、菜单命令等)的消息是不相同的。例如，按钮控件IDC_BUTTON1的消息有两个：BN_CLICKED和BN_DOUBLECLICKED，分别表示当用户单击或双击该按钮时产生的消息。<br />u      一般不需要对图5.10对话框中的[OK](确定)与[Cancel](取消)按钮进行消息映射，因为系统已自动设置了这两个按钮的动作，当用户单击这两个按钮都将自动关闭对话框。 <br />5.1对话框的使用<br />Ø     程序在三处添加新用户代码 <br />     (1)在头文件(MyDlg.h文件)中声明消息处理函数OnButton1：<br />     protected:<br />             //{{AFX_MSG(CMyDlg)<br />            afx_msg void OnButton1();<br />             //}}AFX_MSG<br />     (2)在MyDlg.cpp源文件开头部分的消息映射入口，添加消息映射宏：<br />     BEGIN_MESSAGE_MAP(CMyDlg, CDialog)         // 消息映射开始<br />             //{{AFX_MSG_MAP(CMyDlg)<br />             ON_BN_CLICKED(IDC_BUTTON1, OnButton1)<br />             //}}AFX_MSG_MAP<br />     END_MESSAGE_MAP()                                    // 消息映射结束<br />     表明消息及其处理函数之间的联系。单击按钮控件IDC_BUTTON1时，系统将自动调用OnButton1函数。<br />     (3)在MyDlg.cpp文件中写入一个空的消息处理函数的模板，以便用户填入具体代码，如下面的框架：<br />     void CMyDlg::OnButton1() <br />     {<br />           // TODO: Add your control notification handler code here<br />     }<br />5.1对话框的使用<br />Ø    在程序中使用对话框 <br />     (1)利用项目工作区的ClassView标签项，将Ex_SDIHello.cpp源文件打开；<br />     (2)在Ex_SDIHello.cpp源文件的前面加上包含类CMyDlg的头文件：<br />     #include "MyDlg.h"<br />     (3)在InitInstance函数体中的“return TRUE;”语句之前添加下列代码：<br />     CMyDlg dlg;<br />     dlg.DoModal();<br />     DoModal是模式对话框最常用的函数，它负责对话框的显示和终止。<br />     (4)编译并运行。在程序的一开始就出现用户自己设计的对话框，单击[Button1]按钮将弹出“欢迎进入对话框的设计！”的消息对话框来。<br />5.2控件的创建和使用方法 <br />5.2.1控件的创建方法 <br />     控件的创建方式有以下两种：<br />     1。一种是在对话框模板中用编辑器指定控件，也就是说，将控件的父窗口指定为对话框，这样做的好处是明显的，因为当应用程序启动该对话框时，Windows系统就会为对话框创建控件；而当对话框消失时，控件也随之清除。<br />     2。另一种方式是将控件看作是任一窗口的子窗口，并通过调用相应的Create函数来创建。<br />     [例Ex_SDIHello] 在上例Ex_SDIHello的基础上用子窗口方式创建控件。<br />     (1)打开CMyDlg的头文件MyDlg.h，添加一个按钮类CButton指针变量：<br />     class CMyDlg : public CDialog<br />     {<br />            // Construction<br />            public:<br />             CMyDlg(CWnd* pParent = NULL);   // standard constructor<br />            CButton *m_btnWnd;<br />            …<br />     } <br />5.2控件的创建和使用方法<br />     (2)按Ctrl+W快捷键打开对话框，切换到Message Maps页面，在Object IDs列表中选定“CMyDlg”项，并在Message列表中找到WM_INITDIALOG消息，双击鼠标左键，然后单击[Edit Code]按钮，添加下列代码：<br />     BOOL CMyDlg::OnInitDialog() <br />     {        CDialog::OnInitDialog();<br />           // TODO: Add extra initialization here<br />             m_btnWnd = new CButton();             // 构造按钮控件<br />             m_btnWnd -&gt;Create("你好", WS_CHILD | WS_VISIBLE | BS_PUSHBUTTON,<br />                                CRect(20, 20, 120, 60), this, 201); // 创建<br />            CFont *font = this-&gt;GetFont();            // 获取对话框的字体<br />             m_btnWnd-&gt;SetFont(font);                            // 设置控件字体        <br />            return TRUE;  // return TRUE unless you set the focus to a control<br />                   // EXCEPTION: OCX Property Pages should return FALSE<br />     }<br />     Create用来创建一个按钮控件，第一个参数指定按钮的标题，第二个参数指定控件的风格，第三个参数指定它在父窗口中的位置和大小，第四个参数指定父窗口指针，最后一个参数是指定该控件的标识值。<br />     (3)编译并运行，<br />结论：采用第一种办法 <br />5.2控件的创建和使用方法<br />5.2.2控件的数据交换和数据校验 ddx交换，ddv校验<br />     例如，为CMyDlg类的按钮控件IDC_BUTTON1添加并使用其成员变量m_MyBtn，则其相应的过程如下：<br />     (1)启动MFC ClassWizard，切换到Member Variables页面，如图。<br />     (2)选定Class name为CMyDlg，然后在Control IDs列表中，选定所要关联的控件ID号IDC_BUTTON1，双击鼠标左键或单击[Add Variable]按钮，如图。 <br />5.2控件的创建和使用方法<br />     (3)在Member variable name框中填好与控件相关联的成员变量m_MyBtn，单击[OK]，在Control IDs列表中出现刚才添加的内容。<br />     需要说明的是，<br />u     DDV/DDX技术中，允许为同一个控件关联多个数据成员变量，但必须保证这些变量名是互不相同的（control（控件类），value（具体数据））且这些变量在同一个类型不能有多个变量。<br />u     如果添加的成员变量是一个数值类型，则在MFC ClassWizard对话框的Member Variables页面下方还要求用户输入变量的范围，这就是控件的数据校验设置。<br />u     同样方法为edit添加m_stredit<br />(4)单击[OK]，发现ClassWizard作了以下三方面的修改。<br />u     1在MyDlg.h文件中，添加与控件关联的成员变量的声明，代码如下：<br />    // Dialog Data<br />    //{{AFX_DATA(CMyDlg)<br />    enum { IDD = IDD_DIALOG1 };<br />    CButton  m_MyBtn;<br />    CString  m_strEdit;<br />    //}}AFX_DATA  <br />5.2控件的创建和使用方法<br />u     2在MyDlg.cpp文件中的CMyDlg构造函数实现代码处，添加数据成员变量的初始代码：<br />     CMyDlg::CMyDlg(CWnd* pParent /*=NULL*/)<br />     : CDialog(CMyDlg::IDD, pParent)<br />     {        //{{AFX_DATA_INIT(CMyDlg)<br />             m_strEdit = _T("");<br />             //}}AFX_DATA_INIT<br />     }<br />u     3在MyDlg.cpp文件中的DoDataExchange函数体内，添加控件的DDX/DDV代码。<br />     void CMyDlg::DoDataExchange(CDataExchange* pDX)<br />     {        CDialog::DoDataExchange(pDX);        // 调用此函数作为DDX的开始<br />             //{{AFX_DATA_MAP(CMyDlg)<br />             DDX_Control(pDX, IDC_BUTTON1, m_MyBtn);<br />             DDX_Text(pDX, IDC_EDIT1, m_strEdit);<br />             DDV_MaxChars(pDX, m_strEdit, 20); <br />             //}}AFX_DATA_MAP<br />     }<br />5.2控件的创建和使用方法<br />     (5)为一个控件定义一个关联的数据成员变量后，就可以使用CWnd::UpdateData  控件相关数据的输入和读取<br />True:控件传给变量<br />False：变量传给控件<br />函数实现控件的控制，包括控件相关数据的输入和读取。<br />     运行该程序，当在编辑框中输入“Hello”后，单击[Button1]按钮，则此时该按钮的名称就变成了编辑框控件中的内容“Hello”，如图。 <br />void c123::OnButton1()  c123--button1—click<br />{<br />     // TODO: Add your control notification handler code here<br />UpdateData();<br />m_mybtn.SetWindowText(m_stredit);<br />     <br />}<br />5.2控件的创建和使用方法<br />5.2.3控件的通用属性 <br />5.2控件的创建和使用方法<br />5.2.4控件的消息 <br />     控件的编程和使用过程中，可以向控件发送消息来完成特定的任务，或者是根据控件产生的消息执行自己的代码(主要)。<br />     应用程序创建一般控件或公共控件之后，控件的状态发生改变时，控件就会向其父窗口发送消息，这个消息称为“通知消息”。对于每个消息，系统都会用一个MSG结构来记录，MSG具有下列结构：<br />     typedef struct tagMSG {             // msg  <br />                  HWND           hwnd;                  // 接收到消息的窗口句柄<br />                  UINT          message;        // 消息<br />                  WPARAM wParam;   // 消息的附加信息，它的含义取决于message<br />                  LPARAM         lParam;          // 消息的附加信息，它的含义取决于        message<br />                  DWORD  time; // 消息传送时的时间<br />                  POINT          pt;                // 消息传送时，光标的屏幕坐标<br />     } MSG; <br />     对于一般控件来说，通知消息通常是一条WM_COMMAND消息，这条消息的wParam参数的低位字中含有控件标识符，wParam参数的高位字则为通知代码，lParam参数则是指向控件的句柄。<br />5.2控件的创建和使用方法<br />     对于公共控件，通知消息通常是一条WM_NOTIFY消息，wParam参数是发送通知消息的控件的标识符，lParam参数是指向NMHDR结构的指针。对于有些控件的通知消息，lParam参数是指向一个更大的结构，且结构中的第一个成员还必须具有NMHDR结构。 <br />     对于控件的WM_COMMAND或是WM_NOTIFY消息，都可以用ClassWizard对它们加以映射。 <br />     尽管控件的各自通知消息并不相同，但下列的通知消息是所有Windows控件所共有的：<br />     NM_CLICK                     在控件中单击鼠标左按钮<br />     NM_DBLCLK                   在控件中双击鼠标左按钮<br />     NM_RCLICK                    在控件中单击鼠标左按钮<br />     NM_RDBLCLK                             在控件中双击鼠标右按钮<br />     NM_RETURN                 当控件具有输入焦点时按下ENTER键<br />     NM_SETFOCUS             控件得到输入焦点<br />     NM_KILLFOCUS             控件失去输入焦点<br />     NM_OUTOFMEMORY                 没有足够的内存使控件完成操作 <br />添加消息映射<br />类向导－c123-c123-oncommand:<br /> <br />BOOL c123::OnCommand(WPARAM wParam, LPARAM lParam) <br />{<br />     // TODO: Add your specialized code here and/or call the base class<br />WORD   aa=HIWORD(wParam);<br />WORD        bb=LOWORD(wParam);<br />if((bb==IDC_BUTTON2)&amp;&amp;(aa==BN_CLICKED))<br />MessageBox("abc");          <br />return CDialog::OnCommand(wParam, lParam);<br />}<br /> <br />5.3常用控件 <br />     根据控件的特性和功能，一般可将其分为三类：Windows公共控件、ActiveX控件以及MFC新增的一些控件等。 <br />5.3常用控件<br />5.3.1静态控件和按钮 <br />Ø静态控件 <br />一个静态控件是用来显示一个字符串、框、矩形、图标、位图或增强的图元文件。它可以被用来作为标签、框或用来分隔其它的控件。一个静态控件一般不接收用户输入，也不产生通知消息。<br />在对话框编辑器的控件工具栏中，属于静态控件的有：静态文本(    )、组框(    )和静态图片(添加bmp图片资源－设置图片为bmp格式－选择)(    )三种。 <br />5.3常用控件<br />Ø     按钮 <br />     (1)按键按钮<br />     通常可以立即产生某个动作，执行某个命令，常被称为命令按钮。两种风格：标准按键按钮和默认按键按钮。从外观上来说，默认按键按钮是在标准按键按钮的周围加上一个黑色边框，这个黑色边框表示该按钮已接受到键盘的输入焦点，只须按回车键就能按下该按钮。具体设定的方法是在按键按钮属性对话框的Style页面中选中“Default button”项。<br />     (2)单选按钮<br />     外形是在文本前有一个圆圈，被选中时，就标上一个黑点，可分为一般和自动两种类型。自动类型中，若选中同组按钮中的某个单选按钮，则其余的单选按钮的选中状态就会清除，保证了多个选项始终只有一个被选中。<br />     (3)复选框<br />     外形是在文本前有一个空心方框，被选中时，复选框中就加上一个“ü”标记，三态复选框还有种处于灰框状态，表示“不确定”。设定成三态复选框的方法是在复选框属性对话框的Style页面中选中“Tri-state”项。 <br />5.3常用控件<br />Ø     按钮的消息 <br />     常见的只有两个：BN_CLICKED和BN_DOUBLE- CLICKED。 <br />Ø     按钮选中操作 <br />u       最常用的按钮操作是设置或获取一个或多个按钮的选中状态。CButton类的成员函数SetCheck和GetCheck设置或获取指定按钮的选中状态，原型：<br />     void SetCheck( int nCheck );<br />     int GetCheck( ) const; <br />     nCheck和GetCheck函数返回的值可以是：0表示不选中，1表示选中，2表示不确定(仅用于三态按钮)。 <br />u       对于多个单选按钮的选中状态的设置或获取，需要使用CWnd类的成员函数CheckRadioButton和GetCheckedRadioButton，原型：<br />     void CheckRadioButton( int nIDFirstButton, int nIDLastButton, int nIDCheckButton );<br />     int GetCheckedRadioButton( int nIDFirstButton, int nIDLastButton );<br />     nIDFirstButton和nIDLastButton指定这组单选按钮的第一个和最后一个按钮ID值，nIDCheckButton指定要设置选中状态的按钮ID值，函数GetCheckedRadioButton返回被选中的按钮ID值。  <br />5.3常用控件<br />     [例Ex_Ctrls1] 用静态图片、单选按钮、复选框和按键按钮控件，设计界面。开始，所有单选按钮都是灰显的，不能选择，选中“允许”复选框后，所有单选按钮可以使用，选定一个单选框后，单击[应用]。<br />     (1)用MFC AppWizard(exe)创建一个名为Ex_CommCtrls的基于对话框的应用程序。在创建的第一步，将应用程序类型选择为Dialog Based，按[Finish]。自动显示出该程序的对话框资源模板内容。<br />     (2)打开属性对话框将其标题改为“使用Windows常用控件”。<br />     (3)用编辑器为对话框添加如下表所示的一些控件。<br />     (4)四个单选按钮的Tab次序应连续，从IDC_RADIO1到IDC_RADIO4依次增加。  <br />     (5)按Ctrl+W快捷键，切换到Member Variables页面，在Class name中选择CEx_CommCtrlstDlg，选中复选框控件ID号IDC_CHECK1，双击鼠标或单击Add Variables按钮。 <br />5.3常用控件<br />     (6)切换到ClassWizard的Messsage Maps页面，为复选框IDC_CHECK1和按钮IDC_BUTTON1增加BN_CLICKED的消息映射，添加代码：<br />      void CEx_CommCtrlsDlg::OnCheck1() <br />      {          UpdateData();<br />                for (int i=0; i&lt;4; i++)<br />                          GetDlgItem(IDC_RADIO1 + i)-&gt;EnableWindow(m_bEnabled);<br />      }<br />      void CEx_CommCtrlsDlg::OnButton1() <br />      {          UpdateData();<br />                if (!m_bEnabled) return;<br />                int nID = GetCheckedRadioButton(IDC_RADIO1, IDC_RADIO4);<br />                if (nID == IDC_RADIO1)<br />                {          MessageBox("1");            }<br />                if (nID == IDC_RADIO2)<br />                {          MessageBox("2");            }<br />                if (nID == IDC_RADIO3) <br />                {          MessageBox("3");          }<br />                if (nID == IDC_RADIO4) <br />                {          MessageBox("4");          }<br />      } <br />5.3常用控件<br />     (7)定位到CEx_CommCtrlsDlg::OnInitDialog函数体中，添加下列代码：<br />     BOOL CEx_CommCtrlsDlg::OnInitDialog()<br />     {        CDialog::OnInitDialog();<br />            …<br />             CheckRadioButton(IDC_RADIO1, IDC_RADIO4, IDC_RADIO1); //设置第一个单选按钮为选中<br />             OnCheck1(); －－－why？<br />            return TRUE;  // return TRUE  unless you set the focus to a control<br />     }<br />     (8)编译并运行。<br />5.3常用控件<br />5.3.2编辑框 <br />Ø     编辑框的属性和风格 <br />5.3常用控件<br />Ø      编辑框的基本操作 <br />     (1)口令设置<br />     默认的口令字符是“*”，应用程序可以用成员函数CEdit::SetPasswordChar 来定义自己的口令字符，其函数原型如下：<br />     void SetPasswordChar( TCHAR ch ); <br />     (2)选择文本<br />     用鼠标来选择文本的操作方法：在要选择的文本的一端按下鼠标左键并拖动鼠标，到另一端释放鼠标键。<br />     用键盘来选择文本的方法：在按光标方向移动键的同时，按住Shift键。<br />     应用程序中可以通过编程选择文本，需要通过调用成员函数CEdit::SetSel 来实现。与该函数相对应的还有CEdit::GetSel和CEdit::ReplaceSel，分别用来获取编辑框中前选择的开始和结束的位置以及替换被选择的文本。<br />     (3)设置编辑框的页面边距<br />     设置编辑框的页面边距可以使文本在编辑框显示更具满意效果trim，应用程序可通过调用成员函数CEdit::SetMargins来实现，函数原型：<br />     void SetMargins( UINT nLeft, UINT nRight );<br />     (4)剪帖板操作<br />     编辑框自动支持文本的复制、粘贴、剪切的键盘操作，对应的快捷健为Ctrl+C、Ctrl+V和Ctrl+X。若应用程序调用CEdit::Undo函数时，还可撤消当前的操作，再调用一次该函数，则恢复刚才的操作。   <br />5.3常用控件<br />(5)获取多行编辑框文本<br />     一种是使用DDX/DDV，将编辑框控件所关联的变量类型选定为CString后，则不管多行编辑框的文本有多少都可用此变量来保存。这种方法不能单独获得多行编辑框中的某一行文本。<br />     另一种方法是使用编辑框CEdit类的相关成员函数来获取文本。例如，下面的代码将显示编辑框中第二行的文本内容：<br />     char str[100];－－在头文件定义<br />     if (m_Edit.GetLineCount()&gt;=2)     // 判断多行编辑框的文本是否有两行以上<br />     {      int nChars;<br />            nChars = m_Edit.LineLength(m_Edit.LineIndex(1));        // 获取第二行文本的字符个数<br />     // 0表示第一行，1表示第二行，依次类推。LineIndex用于将文本行转换成<br />     // 能被LineLength识别的索引<br />             m_Edit.GetLine(1,str,nChars);        // 获取第二行文本<br />             str[nChars] = '\0';<br />             MessageBox(str);<br />     } <br />5.3常用控件<br />Ø     编辑框的通知消息<br />     [例Ex_Ctrls2] 使用静态文本、组框、编辑框以及按钮等控件，设计界面。用户在“成绩1”、“成绩2”和“成绩3”编辑框中输入成绩后，单击[计算平均分]按钮，将显示出这三个成绩的平均分。<br />     (1)打开前面的Ex_CommCtrls。<br />     (2)向应用程序中添加一个对话框资源，打开属性对话框将其标题改为“使用编辑框”，ID号改为IDD_EDIT。<br />     (3) 用编辑器为对话框添加如下表所示的一些控件。<br />     (4)双击对话框模板或按Ctrl+W快捷键，为对话框资源IDD_EDIT创建一个对话框类CEditDlg。  <br />5.3常用控件<br />     (5)打开ClassWizard的Member Variables标签，在Class name中选择CEditDlg，选中所需的控件ID号，双击鼠标或单击Add Variables按钮。依次为控件增加成员变量。 <br />     (6)切换到ClassWizard的Messsage Maps标签页，为CEditDlg增加WM_INITDIALOG的消息映射，并添加下列代码：<br />     BOOL CEditDlg::OnInitDialog() <br />     {        CDialog::OnInitDialog();<br />             m_strAve = "0.00";<br />             UpdateData(FALSE);// 将成员变量数据传给控件，并在控件中显示<br />            return TRUE;  <br />     }<br />5.3常用控件<br />     (7)用ClassWziard为按钮IDC_BUTTON1添加BN_CLICKED的消息映射，并增加下列代码：<br />     void CEditDlg::OnButton1() <br />     {        UpdateData();                   // 将控件显示的数据传给成员变量<br />     double ave = (double)(m_nScore1+m_nScore2+m_nScore3)/3.0;<br />     m_strAve.Format("%6.2f", ave);<br />     UpdateData(FALSE);                  // 将成员变量数据传给控件并显示<br />     }<br />     (8)定位到CEx_CommCtrlsDlg::OnButton1函数处，修改代码如下：<br />     if (nID == IDC_RADIO1) {<br />            CEditDlg dlg;<br />             dlg.DoModal();<br />     }<br />     (9)在Ex_CommCtrlsDlg.cpp文件的开始处，增加包含CEditDlg的头文件。<br />     #include         “EditDlg.h”   //目的是“从启动窗体运行本窗体！”<br />     (10)编译运行并测试。<br />如何启动一个新窗体<br />5.3常用控件<br />5.3.3列表框<br />Ø     列表框的风格  <br />5.3常用控件<br />Ø     列表框的基本操作<br />     (1)添加列表项<br />     列表框创建时是一个空的列表，需要添加或插入一些列表项。CListBox类成员函数AddString 和InsertString用来向列表框增加列表项，原型：<br />     int AddString( LPCTSTR lpszItem );<br />     int InsertString( int nIndex, LPCTSTR lpszItem ); <br />     CListBox的SetItemData和SetItemDataPtr能使数据和某个列表项关联起来。<br />     int SetItemData( int nIndex, DWORD dwItemData );<br />     int SetItemDataPtr( int nIndex, void* pData ); <br />     (2)删除列表项 <br />     CListBox类成员函数DeleteString和ResetContent分别用来删除指定的列表项和清除列表框所有项目。它们的函数原型如下：<br />     int DeleteString( UINT nIndex );        // nIndex指定要删除的列表项的索引<br />     void ResetContent( );<br />     若在添加列表项时使用SetItemDataPtr函数，不要忘记在进行删除操作时及时将关联数据所占的内存空间释放出来。<br />5.3常用控件<br />     (3)查找列表项<br />     CListBox类成员函数FindString 和FindStringExact分别用来在列表框中查找所匹配的列表项，其中FindStringExact的查找精度最高。<br />     int FindString( int nStartAfter, LPCTSTR lpszItem ) const;<br />     int FindStringExact( int nIndexStart, LPCTSTR lpszFind ) const;<br />     lpszFind和lpszItem指定要查找的列表项文本，nStartAfter和nIndexStart指定查找的开始位置，若为-1，则从头至尾查找。查到后，这两个函数都将返回所匹配列表项的索引，否则返回LB_ERR。<br />     (4)列表框的单项选择<br />     当选中列表框中某个列表项，可以使用CListBox::GetCurSel来获取这个结果，与该函数相对应的CListBox::SetCurSel 函数是用来设定某个列表项呈选中状态(高亮显示)。<br />     int GetCurSel( ) const;                   // 返回当前选择项的索引<br />     int SetCurSel( int nSelect );<br />     nSelect指定要设置的列表项索引，错误时这两个函数都将返回LB_ERR。<br />     若要获取某个列表项的字符串，可使用下列函数：<br />     int GetText( int nIndex, LPTSTR lpszBuffer ) const;<br />     void GetText( int nIndex, CString&amp; rString ) const;<br />     nIndex指定列表项索引，lpszBuffer 和rString是用来存放列表项文本。 <br />5.3常用控件<br />     (5)列表框的多项选择<br />     在列表框的Style属性对话框中选中多选(Multiple)或扩展多选(Extended)类型后，就可以在列表框中进行多项选择。要想获得选中的多个选项，通过需要用ClassWizrd映射LBN_SELCHANGE消息，添加类似代码：<br />     void CListBoxDlg::OnSelchangeList1() <br />     {      int nCount = m_list.GetSelCount();        // 获取用户选中的项数<br />            if (nCount != LB_ERR)<br />            {        int *buffer = new int[nCount];        // 开辟缓冲区<br />                    m_list.GetSelItems(nCount,buffer);<br />                    CString allStr, str;<br />                    allStr.Empty();                            // 字符串清空<br />                    for (int i=0; i&lt;nCount; i++)<br />                    {        m_list.GetText(buffer[i], str);         <br />                                allStr = allStr + “[” + str + “]”;                                   }<br />                    delete []buffer;                        // 释放内存<br />                    // MessageBox(allStr);                    // 处理获得的文本<br />            }<br />     } <br />5.3常用控件<br />Ø     列表框的通知消息 <br />     [例Ex_Ctrls3] 将一个SCORE结构(含有三门成绩的数据成员)数据和列表框中每一个学生姓名列表关联起来。用户单击[添加记录]按钮时，学生成绩记录中的“姓名”被添加在列表框中，该学生的成绩与该列表项关联。单击[删除记录]时，列表框中当前选择项被删除，相关联的数据所占的内容空间被释放。任何时候选中列表框中某个学生，相应的记录数据都被显示出来。<br />     (1)打开前面创建的基于对话框应用程序程序项目Ex_CommCtrls。<br />     (2)向应用程序中添加一个对话框资源IDD_LISTBOX，标题定为“使用列表框”，用ClassWizard为此对话框类定义为CListBoxDlg。<br />     (3)删除原来的Cancel按钮，用编辑器为对话框添加如下表所示的控件。 <br />5.3常用控件<br />      (4)打开ClassWizard的Member Variables页面，在Class name中选择CListBoxDlg，选中所需的控件ID号，双击鼠标或单击Add Variables按钮。 <br />      (5)切换到ClassWizard的Messsage Maps标签页，为按钮IDC_DATA_ADD添加BN_CLICKED的消息映射，增加代码：<br />      void CListBoxDlg::OnDataAdd() <br />      {          UpdateData(TRUE);<br />                if (m_strName.IsEmpty())                        // 判断m_strName是否为空<br />                {          MessageBox("姓名不能为空！");                         <br />                          return;          }<br />                m_strName.TrimLeft();                // 裁剪m_strName左边的空格<br />                m_strName.TrimRight();              // 裁剪m_strName右边的空格<br />                if ((m_List.FindString(-1,m_strName)) != LB_ERR)<br />                {          MessageBox("列表框中已有相同姓名，不能添加！");                 <br />                          return;          }<br />                int nIndex = m_List.AddString(m_strName);          // 向列表框添加学生姓名<br />                // 将该学生成绩与新增的列表项关联起来          <br />                SCORE data;<br />                data.score1 = m_nScore1;<br />                data.score2 = m_nScore2;<br />                data.score3 = m_nScore3;<br />                m_List.SetItemDataPtr(nIndex, new SCORE(data));  <br />      } <br />5.3常用控件<br />         (6)用ClassWizard为按钮IDC_DATA_DEL添加BN_CLICKED的消息映射，增加代码：<br />      void CListBoxDlg::OnDataDel() <br />      {        int nIndex = m_List.GetCurSel();   // 获得当前选择项的索引<br />                if (nIndex != LB_ERR)<br />                {          m_List.DeleteString(nIndex);     // 删除当前选择项<br />                          m_strName.Empty();<br />                          m_nScore1 = m_nScore2 = m_nScore3 = 0;<br />                          UpdateData(FALSE);<br />                }<br />                else           MessageBox("当前没有选择项或列表框操作失败！");<br />      }<br />      (7)用ClassWizard为按钮IDC_LIST1添加LBN_SELCHANGE的消息映射，增加代码：<br />      void CListBoxDlg::OnSelchangeList1() <br />      {        int nIndex = m_List.GetCurSel();<br />                if (nIndex != LB_ERR) <br />                {          m_List.GetText(nIndex, m_strName);               <br />                          SCORE *data=(SCORE *)m_List.GetItemDataPtr(nIndex);<br />                          m_nScore1 = data-&gt;score1;<br />                          m_nScore2 = data-&gt;score2;<br />                          m_nScore3 = data-&gt;score3;<br />                          UpdateData(FALSE);<br />                }<br />      } <br />5.3常用控件<br />(8)用ClassWizard为对话框添加WM_DESTROY的消息映射，增加代码：<br />     void CListBoxDlg::OnDestroy()         // 此消息是当对话框关闭时发送的<br />     {      for (int nIndex = m_List.GetCount()-1; nIndex&gt;=0; nIndex--)<br />            {// 删除所有与列表项相关联的SCORE结构数据，并释放内存<br />                    delete (SCORE *)m_List.GetItemDataPtr(nIndex);          <br />            }<br />             CDialog::OnDestroy();        // 关闭对话框<br />     }<br />     对话框被清除时发送WM_DESTROY消息。用户在此消息的映射函数中添加一些对象删除代码，以便在对话框清除前有效地释放内存空间。 <br />5.3常用控件<br />     (9)打开ListBoxDlg.h源文件，在类CListBoxDlg定义的public处添加下列代码：<br />     public:<br />     struct SCORE<br />     {      int score1;<br />            int score2;<br />            int score3;<br />     };<br />     (10) 定位到CEx_CommCtrlsDlg::OnButton1函数处，修改代码如下：<br />     if (nID == IDC_RADIO2)<br />     {        CListBoxDlg dlg;<br />             dlg.DoModal();<br />     }<br />     (11)在Ex_CommCtrlsDlg.cpp文件的开始处，增加包含CListBoxDlg的头文件。<br />     #include         “ListBoxDlg.h”<br />     (12) 编译运行并测试。 <br />5.3常用控件<br />5.3.4组合框 <br />Ø     组合框的风格及类型 <br />5.3常用控件<br />Ø      组合框常见操作 <br />      组合框的操作大致分为两类，一类是对组合框中的列表框进行操作，另一类是对组合框中的编辑框进行操作。这些操作都可以调用CComboBox成员函数来实现。<br />Ø      组合框的通知消息 <br />      绘图时还需要下列一些技巧：<br />§      需要更新或重新绘制窗口的外观时，应用程序就会发送WM_PAINT消息。用ClassWizard在对话框中映射WM_PAINT消息的目的是执行绘图代码。<br />§      防止用系统默认的颜色向对话框进行重复绘制，须调用UpdateWindow函数。UpdateWindow是CWnd的无参数的成员函数，目的是绕过系统的消息列队，直接发送或停止发送WM_PAINT消息。窗口没有需要更新的区域时，就停止发送。绘制完图形时，没有WM_PAINT消息的发送，系统不会用默认的颜色对窗口进行重复绘制。<br />§      如果对话框中的任何部分变为无效时，对话框的OnPaint函数都会自动调用。可以通过调用Invalidate函数来通知系统此时的窗口状态已变为无效，强制系统调用WM_PAINT消息函数OnPaint重新绘制。  <br />5.3常用控件<br />     [例Ex_Ctrls4] 根据用户从组合框中选择的填充样式，在对话框中绘制一个矩形区域，如图。 <br />     (1)打开前面创建的基于对话框应用程序程序项目Ex_CommCtrls。<br />     (2)向应用程序中添加一个对话框资源IDD_COMBO，标题定为“使用组合框”，用ClassWizard为此对话框类定义为CComboDlg。<br />     (3) 删除原来的Cancel按钮，用编辑器为对话框添加控件。  <br />5.3常用控件<br />     (4)打开ClassWizard的Member Variables标签，为组合框IDC_PATTERN增加成员变量m_Pattern，其类型为Control类的CComboBox        。<br />     (5)为CComboDlg类添加一个int类型的成员变量m_nDrawPattern。<br />     (6)切换到ClassWizard的Messsage Maps标签页，为组合框IDC_PATTERN添加CBN_SELCHANGE(组合框中的当前选项发生改变)的消息映射，并增加下列代码：<br />     void CComboDlg::OnSelchangePattern() <br />     {      int nIndex = m_Pattern.GetCurSel();// 获得当前选项的索引<br />            if (nIndex != CB_ERR)<br />            {        m_nDrawPattern = m_Pattern.GetItemData(nIndex); <br />                                                     // 获得与当前选项相关联的数据<br />                    Invalidate();  // 强制系统调用OnPaint函数重新绘制<br />            }<br />     } <br />5.3常用控件<br />     (7)用ClassWizard为对话框添加WM_PAINT的消息映射，并增加下列代码：<br />     void CComboDlg::OnPaint() <br />     {        CPaintDC dc(this); // device context for painting(绘图所必须的设备环境)            <br />            CWnd* pWnd = GetDlgItem(IDC_DRAW);// 获得控件IDC_DRAW的窗口指针<br />             pWnd-&gt;UpdateWindow();          // 避免系统自动重绘<br />            CDC* pDC = pWnd-&gt;GetDC();  // 获得所需要的绘图设备环境<br />            CBrush drawBrush;            // 定义一个画刷<br />             drawBrush.CreateHatchBrush(m_nDrawPattern,RGB(0,0,0));// 创建画刷<br />            CBrush* pOldBrush = pDC-&gt;SelectObject(&amp;drawBrush);  <br />                                                    // 将画刷选入当前设备环境中<br />            CRect rcClient;                         // 定义一个CRect变量<br />             pWnd-&gt;GetClientRect(rcClient);         // 获得窗口客户区大小<br />             pDC-&gt;Rectangle(rcClient);         // 用当前画刷绘制一个矩形区域<br />             pDC-&gt;SelectObject(pOldBrush);        // 恢复设备环境原来的画刷设置<br />} <br />5.3常用控件<br />     (8)用ClassWizard为对话框添加WM_INITDIALOG的消息映射，增加代码：<br />     BOOL CComboDlg::OnInitDialog() <br />     {        CDialog::OnInitDialog();<br />            CString str[6] = {"水平线","竖直线","向下斜线","向上斜线","十字线","交叉线"};<br />            int nIndex;<br />            for (int i=0; i&lt;6; i++)<br />            {<br />                    nIndex = m_Pattern.AddString(str[i]);<br />                    m_Pattern.SetItemData(nIndex,i);<br />            }<br />             m_Pattern.SetCurSel(0);<br />             m_nDrawPattern = 0;<br />            return TRUE;  <br />     }<br />5.3常用控件<br />     (9)定位到CEx_CommCtrlsDlg::OnButton1函数处，修改代码如下：<br />     if (nID == IDC_RADIO3)<br />     {<br />             CComboDlg dlg;<br />             dlg.DoModal();<br />     }<br />     (10)在Ex_CommCtrlsDlg.cpp文件的开始处，增加包含CComboDlg的头文件。<br />     #include         “ComboDlg.h”<br />    (11)编译运行并测试。<br />复习<br />1 静态控件，按钮，编辑框，列表框，组合框<br />2 如何从一个窗体切换到另外一个窗体<br />3 组合框的使用方法<br />5.3常用控件<br />5.3.5滚动条和进展条 <br />Ø    滚动条 <br />(1)滚动条的基本操作 <br />u       在MFC的CScrollBar类中，函数SetScrollRange是用来设置滚动条的滚动范围的，其原型如下：<br />     SetScrollRange( int nMinPos, int nMaxPos, BOOL bRedraw = TRUE ); <br />u       在CScrollBar类中，设置滚动块位置操作是由SetScrollPos函数来完成的，其原型如下：<br />     int SetScrollPos( int nPos, BOOL bRedraw = TRUE ); <br />u       与SetScrollRange 和SetScrollPos相对应的两个函数是分别用来获取滚动条的当前范围以及当前滚动位置：<br />     void GetScrollRange( LPINT lpMinPos, LPINT lpMaxPos ) ;<br />     int GetScrollPos(); <br />u       需要说明的是：在CScrollBar类的成员函数中，还可以用SetScrollInfo和GetScrollInfo来代替上面提到的四个函数。与前面的函数相比，使用SetScrollInfo函数还能使滚动块的大小随内容的多少而改变。<br />5.3常用控件<br />(2) WM_HSCROLL或WM_VSCROLL消息 <br />     滚动条进行操作时，会向父窗口发送WM_HSCROLL或WM_VSCROLL消息。通过ClassWziard在其对话框中进行映射，产生相应的消息映射函数OnHScroll和OnVScroll，函数原型：<br />     afx_msg void OnHScroll( UINT nSBCode, UINT nPos, CScrollBar* pScrollBar );<br />      afx_msg void OnVScroll( UINT nSBCode, UINT nPos, CScrollBar* pScrollBar ); <br />5.3常用控件<br />Ø    进展条 <br />     (1)进展条的风格<br />      打开进展条的属性对话框。Border指定进展条是否有边框，Vertical指定进展是水平还是垂直的。Smooth表示平滑地填充进展条。 <br />      (2)进展条的基本操作<br />      基本操作：设置范围、当前位置、增量等。通过相关的CProgressCtrl类成员函数实现。<br />      int SetPos( int nPos );<br />      int GetPos();<br />      这两个函数分别用来设置和获取进展条的当前位置。 <br />      void SetRange( short nLower, short nUpper );<br />      void SetRange32(int nLower, int nUpper );<br />      void GetRange( int &amp; nLower, int&amp; nUpper );<br />      分别用来设置和获取进展条范围的上限和下限值。<br />      int SetStep( int nStep );<br />      该函数用来设置进展条的步长并返回原来的步长。<br />      int StepIt();<br />      该函数将当前位置向前移动一个步长并重画进展条以反映新的位置。<br />5.3常用控件<br />5.3.6旋转按钮控件和滑动条 <br />Ø     旋转按钮控件 <br />     (1)旋转按钮控件常用的风格 <br />5.3常用控件<br />(2)旋转按钮控件的基本操作 <br />     成员函数SetBase是用来设置其基数（10或者16近位制）的，函数的原型如下：<br />     int SetBase( int nBase ); <br />     成员函数SetPos和SetRange设置旋转按钮控件的当前位置和范围，它们的函数原型如下：<br />     int SetPos( int nPos );<br />     void SetRange( int nLower, int nUpper ); <br />     成员函数SetAccel和GetAccel用来设置和获取旋转按钮控件的加速度。其中函数SetAccel的原型如下：<br />     BOOL SetAccel( int nAccel, UDACCEL* pAccel ); <br />     参数nAccel表示由pAccel指定的UDACCEL结构的数目。pAccel指向一个UDACCEL结构数组的指针，该数组包含了加速信息，其结构如下：<br />     typedef struct { <br />                    UINT         nSec;         // 位置改变前所等待的秒数。<br />                   UINT        nInc;         // 位置增量。<br />     }UDACCEL, FAR *LPUDACCEL;<br />(3)旋转按钮控件的通知消息<br />     旋转按钮控件的通知消息只有一个：UDN_DELTAPOS，它是当控件的当前数值将要改变时向其父窗口发送的。 <br />5.3常用控件<br />Ø    滑动条 <br />(1)滑动条的风格 <br />5.3常用控件<br />(2)滑动条的基本操作<br />u       成员函数SetPos和SetRange用来设置滑动条的位置和范围，原型如下：<br />     void SetPos( int nPos );<br />     void SetRange( int nMin, int nMax, BOOL bRedraw = FALSE ); <br />u       成员函数SetTic用来设置滑动条控件中的一个刻度线的位置。函数成功调用后返回非零值；否则返回0。函数原型如下：<br />     BOOL SetTic( int nTic ); <br />u       成员函数SetTicFreq设置显示在滑动条中的刻度线的疏密程度。原型如下：<br />     void SetTicFreq( int nFreq );<br />u       成员函数ClearTics用来从滑动条控件中删除当前的刻度线。函数原型如下：<br />     void ClearTics( BOOL bRedraw = FALSE );<br />u       成员函数SetSelection用来设置一个滑动条控件中当前选择的开始和结束位置。其函数原型如下：<br />     void SetSelection( int nMin, int nMax );<br />(3)滑动条的通知消息<br />     消息代码都来自于WM_HSCROLL或WM_VSCROLL消息。  <br />5.3常用控件<br />     [例Ex_Ctrls5]用滚动条、滑动条和旋转按钮控件分别来调整RGB的三个颜色分量，根据指定的颜色填充一个矩形区域。<br />     (1)打开前面创建的基于对话框应用程序程序项目Ex_CommCtrls。<br />     (2)向应用程序中添加一个对话框资源IDD_SCROLL，标题定为“使用滚动条、旋转按钮控件和滑动条”，用ClassWizard为此对话框类定义为CScrollDlg。<br />     (3)删除原来的Cancel按钮，用编辑器为对话框添加如下表所示的一些控件。<br />5.3常用控件<br />     (4)打开ClassWizard的Member Variables标签，在Class name中选择CScrollDlg，选中所需的控件ID号，双击鼠标或单击Add Variables按钮。依次为下列控件增加成员变量。<br />     (5)切换到ClassWizard的Messsage Maps标签页，为编辑框IDC_EDIT1、IDC_EDIT2和IDC_EDIT3添加EN_CHANGE的消息映射，使用它们的消息处理函数名都设为OnChangeEdit，并增加下列代码：<br />     void CScrollDlg::OnChangeEdit() <br />     {      if (!m_bEditOK) return;<br />             UpdateData();<br />             m_Scroll.SetScrollPos(m_RValue);<br />             m_Slider.SetPos(m_GValue);<br />            Draw();<br />     }  <br />5.3常用控件<br />(6)用ClassWizard为对话框添加WM_PAINT的消息映射，增加代码：<br />      void CScrollDlg::OnPaint() <br />      {        CPaintDC dc(this); // device context for painting        <br />                CWnd* pWnd=GetDlgItem(IDC_DRAW);<br />                pWnd-&gt;UpdateWindow();<br />                Draw();<br />      }<br />(7)用ClassWizard为对话框添加WM_INITDIALOG的消息映射，增加代码：<br />      BOOL CScrollDlg::OnInitDialog() <br />      {          CDialog::OnInitDialog();<br />                // 设置滚动条和滑动条的范围和当前位置<br />                m_Scroll.SetScrollRange(0,255);<br />                m_Scroll.SetScrollPos(m_RValue);<br />                m_Slider.SetRange(0,255);<br />                m_Slider.SetPos(m_GValue);<br />                // 设置旋转按钮的范围<br />                m_Spin.SetRange(0,255);<br />                UpdateData(FALSE);          // 将数据传给控件<br />                m_bEditOK = TRUE;<br />                return TRUE; <br />      } <br />5.3常用控件<br />      (8)用ClassWizard为对话框添加WM_HSCROLL的消息映射，增加代码：<br />      void CScrollDlg::OnHScroll(UINT nSBCode, UINT nPos, CScrollBar* pScrollBar) <br />      {        int nID = pScrollBar-&gt;GetDlgCtrlID();<br />                if (nID == IDC_SLIDER1)           // 是滑动条产生水平滚动消息<br />                {          m_GValue = m_Slider.GetPos();}// 获得滑动条当前的位置          }<br />                if (nID == IDC_SCROLLBAR1)    // 是滚动条条产生水平滚动消息<br />                {          switch(nSBCode)<br />                          {          case SB_LINELEFT:m_RValue--;          break;<br />                                          case SB_LINERIGHT:m_RValue++;          break;<br />                                          case SB_PAGELEFT:          m_RValue -= 10;          break;<br />                                          case SB_PAGERIGHT:m_RValue += 10;     break;<br />                                          case SB_THUMBTRACK:m_RValue = nPos;break;<br />                          }<br />                          if (m_RValue&lt;0) m_RValue = 0;<br />                          if (m_RValue&gt;255) m_RValue = 255;<br />                          m_Scroll.SetScrollPos(m_RValue);<br />                }<br />      UpdateData(FALSE);<br />      Draw();                                                                                            <br />      CDialog::OnHScroll(nSBCode, nPos, pScrollBar);<br />      } <br />5.3常用控件<br />     (9)将项目工作区切换到ClassView页面，用鼠标右击“CScrollDlg”项，在弹出的快捷菜单中选中“Add Member Function”，将要增加的函数名设为Draw，函数类型为void，并添加下列代码：<br />     void CScrollDlg::Draw()<br />     {      CWnd* pWnd = GetDlgItem(IDC_DRAW);<br />            CDC* pDC = pWnd-&gt;GetDC();  // 获得窗口当前的设备环境指针<br />            CBrush drawBrush;            // 定义画刷变量<br />             drawBrush.CreateSolidBrush(RGB(m_RValue,m_GValue,m_BValue));<br />            // 创建一个填充色画刷。<br />            // 颜色分量转换成一个32位的RGB颜色值<br />            CBrush* pOldBrush = pDC-&gt;SelectObject(&amp;drawBrush);<br />            CRect rcClient;<br />             pWnd-&gt;GetClientRect(rcClient);<br />             pDC-&gt;Rectangle(rcClient);<br />             pDC-&gt;SelectObject(pOldBrush);<br />     } <br />5.3常用控件<br /> (10)为CScrollDlg类添加一个BOOL型成员变量m_bEditOK，并在其构造函数中将其初值设为FALSE。<br /> (11)定位到CEx_CommCtrlsDlg::OnButton1函数处，修改代码如下：<br />     if (nID == IDC_RADIO4) <br />     {        CScrollDlg dlg;<br />             dlg.DoModal();<br />     }<br /> (12)在Ex_CommCtrlsDlg.cpp文件的开始处，增加包含CScrollDlg的头文件。<br />     #include         “ScrollDlg.h”<br /> (13)编译运行并测试。<br />     在同时使用编辑框和旋转按钮控件时，有时会出现运行错误。为了解决这个问题，本例使用一个办法，即使用一个BOOL型成员变量m_bEditOK作为控制条件，使得所有初始化代码(包括设置旋转控件范围)执行后，才允许编辑框工作。<br />     由于滚动条和滑动条等许多控件都能产生WM_HSCROLL或WM_VSCROLL消息，因此当它们是处在同一方向(水平或垂直)时，就需要添加相应代码判断消息是谁产生的。<br />     由于滚动条中间的滚动块在默认时是不会停止在用户操作的位置处的，因此需要调用SetScrollPos函数来进行相应位置的设定。 <br />5.3常用控件<br />5.3.7图像列表和标签控件 <br />Ø     图像列表控件 <br />(1)图像列表的创建<br />     由于Create函数的重载很多，故这里给出最常用的一个原型：<br />     BOOL Create( int cx, int cy, UINT nFlags, int nInitial, int nGrow ); <br />(2)图像列表的基本操作<br />     常见的图像列表的基本操作有：增加、删除和绘制等，成员函数：<br />     int Add( CBitmap* pbmImage, CBitmap* pbmMask );<br />     int Add( CBitmap* pbmImage, COLORREF crMask );<br />     int Add( HICON hIcon ); <br />u       BOOL Remove( int nImage );<br />     该函数用来从图像列表中删除一个由nImage指定的图像，<br />u       BOOL Draw( CDC* pdc, int nImage, POINT pt, UINT nStyle );<br />     该函数用来在由pt指定的位置处绘制一个图像。 <br />u       HICON ExtractIcon( int nImage );<br />     该函数用来将nImage指定的图像扩展为图标。<br />u       COLORREF SetBkColor( COLORREF cr );<br />     该函数用来设置图像列表的背景色，    <br />5.3常用控件<br />Ø     标签控件 <br />(1)标签控件的风格<br />5.3常用控件<br />(2)标签控件的基本操作 <br />u       GetItemRect是用来获取标签的边界大小，原型如下：<br />      CSize SetItemSize( CSize size );           <br />      void SetPadding( CSize size );            <br />      int SetMinTabWidth( int cx );           <br />      BOOL GetItemRect( int nItem, LPRECT lpRect ) const; <br />u       成员函数InsertItem、DeleteItem表示在一个标签控件中插入、删除某一标签项，DeleteAllItems删除所有的项。原型如下：<br />      BOOL InsertItem( int nItem, LPCTSTR lpszItem );<br />      BOOL InsertItem( int nItem, LPCTSTR lpszItem, int nImage);<br />      BOOL DeleteItem( int nItem );<br />      BOOL DeleteAllItems( ); <br />u       标签控件中使用图像列表时，必须调用CTabCtrl::SetImageList函数来指定一个已创建的图像列表。函数原型如下：<br />      CImageList * SetImageList( CImageList * pImageList ); <br />u       成员函数SetCurSel和GetCurSel用来设置和获取当前选择的标签项，原型如下：<br />      int SetCurSel( int nItem ); <br />      int GetCurSel( ) const; <br />u       成员函数DeselectAll用来重新设置一个标签控件中的项，清除任何被按下的项，函数HighlightItem是使某一个标签项处于高亮状态。原型如下：<br />      void DeselectAll( BOOL fExcludeFocus );<br />      BOOL HighlightItem( int idItem, BOOL fHighlight = TRUE );<br />(3)标签控件的通知消息<br />5.3常用控件<br />   [例Ex_Ctrls6] 使用无模式对话框来构造某个标签页面，一开始将所有的标签页面创建好，然后根据所选择的当前标签选项决定哪个页面需要显示，哪个页面需要隐藏和禁用。<br />     (1)用MFC AppWizard(exe)创建一个基于对话框的应用程序项目Ex_TabCtrl。<br />     (2)打开属性对话框将对话框标题改为“使用标签控件”。<br />     (3) 添加一个默认标签控件IDC_TAB1，调整其大小和布局。<br />     (4)按Ctrl+W，打开ClassWizard对话框，切换到Member Variables页面，在Class name中选择CEx_TabCtrlDlg，选中标签控件ID号IDC_TAB1，双击鼠标或单击Add Variables按钮，为其添加一个CTabCtrl类型的成员变量m_Tab。<br />     (5)添加三个对话框资源IDD_DIALOG1、IDD_DIALOG2和IDD_DIALOG3，并用ClassWizard依次为此三个对话框类定义为CTab1Dlg、CTab2Dlg和CTab3Dlg。 <br />5.3常用控件<br />     (6)将上述三个对话框的属性设为Child Style和None Border(其他默认)，且将对话框尽量缩小，如图。 <br />5.3常用控件<br />     (7) 为三个对话框添加如下表所示的一些控件。 <br />5.3常用控件<br />     (8)在ClassView中的CEx_TabCtrlDlg类名处右击鼠键，从弹出的快捷菜单中选择Add Member Variables，添加下列数据成员：<br />     public:<br />     CTab1Dlg              *m_pTab1Dlg;<br />     CTab2Dlg               *m_pTab2Dlg;<br />     CTab3Dlg              *m_pTab3Dlg;<br />     CImageList            m_ImageList;<br />     (9)打开Ex_TabCtrlDlg.h文件，在class CEx_TabCtrlDlg前添加下列头文件：<br />     #include "Tab1Dlg.h"<br />     #include "Tab2Dlg.h"<br />     #include "Tab3Dlg.h" <br />5.3常用控件<br />(10)在ClassView中的CEx_TabCtrlDlg类名处右击鼠键，选择Add Member Function，添加成员函数SetDlgState和DoTab，并增加下列代码：<br />      void CEx_TabCtrlDlg::SetDlgState(CWnd *pWnd, BOOL bShow)<br />      {          pWnd-&gt;EnableWindow(bShow);               <br />                if (bShow) <br />                {          pWnd-&gt;ShowWindow(SW_SHOW);<br />                          pWnd-&gt;CenterWindow();          // 居中显示<br />                }<br />                else          pWnd-&gt;ShowWindow(SW_HIDE);<br />      }<br />      void CEx_TabCtrlDlg::DoTab(int nTab)<br />      {        if (nTab&gt;2) nTab = 2;             // 确定nTab值不能超过范围<br />                if (nTab&lt;0) nTab = 0;<br />                BOOL bTab[3];<br />                bTab[0] = bTab[1] = bTab[2] = FALSE;<br />                bTab[nTab] = TRUE;                                  <br />                // 切换对话框的显示和隐藏<br />                SetDlgState(m_pTab1Dlg, bTab[0]);<br />                SetDlgState(m_pTab2Dlg, bTab[1]);<br />                SetDlgState(m_pTab3Dlg, bTab[2]);<br />      } <br />5.3常用控件<br />     (11)用ClassWizard为IDC_TAB1映射TCN_SELCHANGE消息，添加代码：<br />     void CEx_TabCtrlDlg::OnSelchangeTab1(NMHDR* pNMHDR, LRESULT* pResult) <br />     {      int nSelect = m_Tab.GetCurSel();<br />            if (nSelect &gt;= 0)<br />                    DoTab(nSelect);<br />            *pResult = 0;<br />     }<br />    (12)按下Ctrl+R快捷键，单击Import按钮，从外部磁盘中调入三个图标文件，并取默认的图标ID号IDI_ICON1、IDI_ICON2和IDI_ICON3。 <br />5.3常用控件<br />(13)在CEx_TabCtrlDlg::OnInitDialog中添加下列代码：<br />      BOOL CEx_TabCtrlDlg::OnInitDialog()<br />      {          CDialog::OnInitDialog();<br />                ……<br />                m_pTab1Dlg = new CTab1Dlg();          // 为无模式对话框分配空间<br />                m_pTab2Dlg = new CTab2Dlg();<br />                m_pTab3Dlg = new CTab3Dlg();<br />                m_pTab1Dlg-&gt;Create(IDD_DIALOG1,&amp;m_Tab);               <br />                m_pTab2Dlg-&gt;Create(IDD_DIALOG2,&amp;m_Tab);<br />                m_pTab3Dlg-&gt;Create(IDD_DIALOG3,&amp;m_Tab);<br />                m_ImageList.Create(16, 16, ILC_COLOR|ILC_MASK, 3, 0);          // 创建图像列表<br />                m_ImageList.Add(AfxGetApp()-&gt;LoadIcon(IDI_ICON1));           m_ImageList.Add(AfxGetApp()-&gt;LoadIcon(IDI_ICON2));<br />                m_ImageList.Add(AfxGetApp()-&gt;LoadIcon(IDI_ICON3));<br />                m_Tab.SetImageList(&amp;m_ImageList);          // 设置TAB控件所使用的图像列表<br />                m_Tab.InsertItem(0, "基本信息", 0);<br />                m_Tab.InsertItem(1, "成绩", 1);<br />                m_Tab.InsertItem(2, "备注", 2);<br />                m_Tab.SetMinTabWidth(80);               // 设置标签项的最小宽度<br />                m_Tab.SetPadding(CSize(12,3));          // 设置标签项和图标周围的间隔<br />                m_Tab.SetCurSel(0);<br />                DoTab(0);<br />                return TRUE;<br />      } <br />5.3常用控件<br />      (14)用ClassWizard为CEx_TabCtrlDlg类添加WM_DESTROY的消息映射，CEx_TabCtrlDlg对话框退出时，删除分配给无模式对话框的内存。代码：<br />     void CEx_TabCtrlDlg::OnDestroy() <br />     {        CDialog::OnDestroy();<br />            if (m_pTab1Dlg) delete m_pTab1Dlg;<br />            if (m_pTab2Dlg) delete m_pTab2Dlg;<br />            if (m_pTab3Dlg) delete m_pTab3Dlg; <br />     }<br />    (15)用ClassWizard为CTab1lDlg类添加WM_INITDIALOG消息的映射，增加下列代码：<br />     BOOL CTab1Dlg::OnInitDialog() <br />     {        CDialog::OnInitDialog();<br />             CheckRadioButton(IDC_RADIO_MAN_1,IDC_RADIO_WOMAN_1,IDC_RADIO_MAN_1);<br />            return TRUE;  // return TRUE unless you set the focus to a control<br />                   // EXCEPTION: OCX Property Pages should return FALSE<br />     }<br />     (16) 编译并运行。<br />5.3常用控件<br />     需要说明的是：<br />§      标签页面的切换可能还不止一种方法。例如，可以先将各个页面的控件与标签控件一起事先全部地添加到对话框模板中，然后再根据标签控件的当前选项确定控件的显示和隐藏，从而达到切换的效果。<br />§      在标签页面控件操作过程中，当按Enter键或Esc键，该页面的对话框就会退出，标签页面变成了一个空白，影响交互效果。解决这个问题的一个办法是在各页面的对话框中屏蔽掉Enter键和Esc键的按键消息，它是通过添加PreTranslateMessage函数的重载来实现的，如下面的代码：<br />     BOOL CTab1Dlg::PreTranslateMessage(MSG* pMsg) <br />     {      if (pMsg-&gt;message == WM_KEYDOWN)<br />            {        if (pMsg-&gt;wParam == VK_RETURN)        return TRUE;                   if (pMsg-&gt;wParam == VK_ESCAPE)        return TRUE;        } <br />            return CDialog::PreTranslateMessage(pMsg);<br />     }<br />5.4通用对话框和消息对话框 <br />5.4.1通用对话框 <br />     Windows提供了一组标准用户界面对话框，它们都有相应的MFC库中的类来支持。所有这些通用对话框类都是从一个公共的基类CCommonDialog派生而来。表列出了这些通用对话框。 <br />CFileDialog dlg(TRUE,"bmp","bmp");<br />if(dlg.DoModal()==IDOK){<br />CFile file;<br />VERIFY(file.Open(dlg.GetPathName(),CFile::modeRead));<br />}   <br />}<br /> <br />5.4通用对话框和消息对话框<br />5.4.2消息对话框 <br />     是最简单的一类对话框，只是用来显示信息的。MFC类库中提供相应的函数实现这样的功能，只要在程序任何地方调用它们。函数原型：<br />     int AfxMessageBox( LPCTSTR lpszText, UINT nType = MB_OK, UINT nIDHelp = 0 );<br />     int MessageBox( LPCTSTR lpszText, LPCTSTR lpszCaption = NULL, UINT nType = MB_OK ); <br />MessageBox("啊啊","aa",MB_ICONQUESTION|MB_OK);</p>
		<p>
				<br />本文来自CSDN博客，转载请标明出处：<a href="http://blog.csdn.net/lxf20054658/archive/2008/12/24/3598645.aspx">http://blog.csdn.net/lxf20054658/archive/2008/12/24/3598645.aspx</a></p>
<img src ="http://www.cppblog.com/cs569652894/aggbug/145785.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/cs569652894/" target="_blank">CrackerCS</a> 2011-05-06 00:45 <a href="http://www.cppblog.com/cs569652894/archive/2011/05/06/145785.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>MFC控件笔记</title><link>http://www.cppblog.com/cs569652894/archive/2011/05/06/145783.html</link><dc:creator>CrackerCS</dc:creator><author>CrackerCS</author><pubDate>Thu, 05 May 2011 16:41:00 GMT</pubDate><guid>http://www.cppblog.com/cs569652894/archive/2011/05/06/145783.html</guid><wfw:comment>http://www.cppblog.com/cs569652894/comments/145783.html</wfw:comment><comments>http://www.cppblog.com/cs569652894/archive/2011/05/06/145783.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/cs569652894/comments/commentRss/145783.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/cs569652894/services/trackbacks/145783.html</trackback:ping><description><![CDATA[
		<p>VC学习笔记1：按钮的使能与禁止<br />用ClassWizard的Member Variables为按钮定义变量，如：m_Button1；<br />则<br />m_Button1.EnableWindow(true); 使按钮处于允许状态<br />m_Button1.EnableWindow(false); 使按钮被禁止，并变灰显示<br />VC学习笔记2：控件的隐藏与显示<br />用CWnd类的函数BOOL ShowWindow(int nCmdShow)可以隐藏或显示一个控件。<br /><br />例1：<br />CWnd *pWnd;<br />pWnd = GetDlgItem( IDC_EDIT1 );     //获取控件指针，IDC_EDIT为控件ID号<br />pWnd-&gt;ShowWindow( SW_HIDE );     //隐藏控件<br /><br />例2：<br />CWnd *pWnd;<br />pWnd = GetDlgItem( IDC_EDIT1 );     //获取控件指针，IDC_EDIT为控件ID号<br />pWnd-&gt;ShowWindow( SW_SHOW );     //显示控件<br /><br />以上方法常用于动态生成控件，虽说用控件的Create函数可以动态生成控件，但这种控件很不好控制，所以用隐藏、显示方法不失为一种替代手段。<br /><br />VC学习笔记3：改变控件的大小和位置<br />用CWnd类的函数MoveWindow()或SetWindowPos()可以改变控件的大小和位置。<br /><br />void MoveWindow(int x,int y,int nWidth,int nHeight);<br />void MoveWindow(LPCRECT lpRect);<br />第一种用法需给出控件新的坐标和宽度、高度；<br />第二种用法给出存放位置的CRect对象；<br />例：<br />CWnd *pWnd;<br />pWnd = GetDlgItem( IDC_EDIT1 );     //获取控件指针，IDC_EDIT1为控件ID号<br />pWnd-&gt;MoveWindow( CRect(0,0,100,100) );     //在窗口左上角显示一个宽100、高100的编辑控件<br /><br />SetWindowPos()函数使用更灵活，多用于只修改控件位置而大小不变或只修改大小而位置不变的情况：<br />BOOL SetWindowPos(const CWnd* pWndInsertAfter,int x,int y,int cx,int cy,UINT nFlags);<br />第一个参数我不会用，一般设为NULL;<br />x、y控件位置；cx、cy控件宽度和高度；<br />nFlags常用取值：<br />SWP_NOZORDER：忽略第一个参数；<br />SWP_NOMOVE：忽略x、y，维持位置不变；<br />SWP_NOSIZE：忽略cx、cy，维持大小不变；<br />例：<br />CWnd *pWnd;<br />pWnd = GetDlgItem( IDC_BUTTON1 );     //获取控件指针，IDC_BUTTON1为控件ID号<br />pWnd-&gt;SetWindowPos( NULL,50,80,0,0,SWP_NOZORDER | SWP_NOSIZE );     //把按钮移到窗口的(50,80)处<br />pWnd = GetDlgItem( IDC_EDIT1 );<br />pWnd-&gt;SetWindowPos( NULL,0,0,100,80,SWP_NOZORDER | SWP_NOMOVE );     //把编辑控件的大小设为(100,80)，位置不变<br />pWnd = GetDlgItem( IDC_EDIT1 );<br />pWnd-&gt;SetWindowPos( NULL,0,0,100,80,SWP_NOZORDER );     //编辑控件的大小和位置都改变<br />以上方法也适用于各种窗口。<br /><br />VC学习笔记4：什么时候设定视中控件的初始尺寸？<br />我在CFormView的视中加入了一个编辑控件，在运行时使它充满客户区，当窗口改变大小时它也跟着改变。<br />改变控件尺寸可以放在On<wbr>Draw()函数中，也可放在CalcWindowRect()函数中，当窗口尺寸发生变化时，它们都将被执行，且CalcWindowRect()函数先于On<wbr>Draw()函数，下例是在CalcWindowRect()函数中修改控件尺寸。<br />重载VIEW类的CalcWindowRect函数，把设定控件的尺寸的语句加入这个函数中。<br />例：<br />void CMyEditView::CalcWindowRect(LPRECT lpClientRect, UINT nAdjustType) <br />{<br />     // TODO: Add your specialized co<wbr>de here and/or call the base class<br /><br />     CFrameWnd *pFrameWnd=GetParentFrame(); //获取框架窗口指针<br /><br />     CRect rect;<br />     pFrameWnd-&gt;GetClientRect(&amp;rect); //获取客户区尺寸<br /><br />     CWnd *pEditWnd=GetDlgItem(IDC_MYEDIT); //获取编辑控件指针，IDC_MYEDIT为控件ID号<br />     pEditWnd-&gt;SetWindowPos(NULL,0,0,rect.right,rect.bottom-50,SWP_NOMOVE | SWP_NOZORDER); //设定控件尺寸，bottom-50是为了让出状态条位置。<br /><br />     CFormView::CalcWindowRect(lpClientRect, nAdjustType);<br />}<br /><br />VC学习笔记5：单选按钮控件（Ridio Button）的使用<br />一、对单选按钮进行分组：<br />每组的第一个单选按钮设置属性：Group，Tabstop，Auto;其余按钮设置属性Tabstop，Auto。<br /><br />如：<br />Ridio1、Ridio2、Ridio3为一组，Ridio4、Ridio5为一组<br /><br />设定Ridio1属性：Group，Tabstop，Auto<br />设定Ridio2属性：Tabstop，Auto<br />设定Ridio3属性：Tabstop，Auto<br /><br />设定Ridio4属性：Group，Tabstop，Auto<br />设定Ridio5属性：Tabstop，Auto<br /><br />二、用ClassWizard为单选控件定义变量，每组只能定义一个。如：m_Ridio1、m_Ridio4。<br /><br />三、用ClassWizard生成各单选按钮的单击消息函数，并加入内容：<br /><br />void CWEditView::On<wbr>Radio1() <br />{<br />     m_Ridio1 = 0;     //第一个单选按钮被选中<br />}<br /><br />void CWEditView::On<wbr>Radio2() <br />{<br />     m_Ridio1 = 1;     //第二个单选按钮被选中<br />}<br /><br />void CWEditView::On<wbr>Radio3() <br />{<br />     m_Ridio1 = 2;     //第三个单选按钮被选中<br />}<br /><br />void CWEditView::On<wbr>Radio4() <br />{<br />     m_Ridio4 = 0;     //第四个单选按钮被选中<br />}<br /><br />void CWEditView::On<wbr>Radio5() <br />{<br />     m_Ridio4 = 1;     //第五个单选按钮被选中<br />}<br /><br />四、设置默认按钮：<br />在定义控件变量时，ClassWizard在构造函数中会把变量初值设为-1，只需把它改为其它值即可。<br />如：<br />//{{AFX_DA<wbr>TA_INIT(CWEditView)<br />m_Ridio1 = 0;     //初始时第一个单选按钮被选中<br />m_Ridio4 = 0;     //初始时第四个单选按钮被选中<br />//}}AFX_DA<wbr>TA_INIT<br /><br /><br />VC学习笔记6：旋转控件（Spin）的使用<br /><br />当单击旋转控件上的按钮时，相应的编辑控件值会增大或减小。其设置的一般步骤为：<br />一、在对话框中放入一个Spin控件和一个编辑控件作为Spin控件的伙伴窗口，<br />设置Spin控件属性：Auto buddy、Set buddy integer、Arrow keys<br />设置文本控件属性：Number<br /><br />二、用ClassWizard为Spin控件定义变量m_Spin，为编辑控件定义变量m_Edit，定义时注意要把m_Edit设置为int型。<br /><br />三、在对话框的On<wbr>InitDialog()函数中加入语句：<br />BOOL CMyDlg::On<wbr>InitDialog() <br />{<br />     CDialog::On<wbr>InitDialog();<br />    <br />     m_Spin.SetBuddy( GetDlgItem( IDC_EDIT1 ) );     //设置编辑控件为Spin控件的伙伴窗口<br />     m_Spin.SetRange( 0, 10 );     //设置数据范围为0-10<br />     return TRUE;<br />}<br /><br />四、用ClassWizard为编辑控件添加EN_CHANGE消息处理函数，再加入语句：<br />void CMyDlg::On<wbr>ChangeEdit1() <br />{<br />     m_Edit = m_Spin.GetPos();     //获取Spin控件当前值<br />}<br />五、将SPIN控件的 “Set Bueey Integer”设置为TRUE。<br />OK!<br /><br /><br />VC学习笔记7：程序结束时保存文件问题<br /><br />在文档-视图结构中，用串行化自动保存文件在各种VC书上都有介绍。现在的问题是我不使用串行化，而是自己动手保存，当点击窗口的关闭按钮时，如何提示并保存文档。<br /><br />用ClassWizard在文档类（CxxDoc）中添加函数CanCloseFrame()，再在其中加入保存文件的语句就可以了。<br />注：要保存的数据应放在文档类（CxxDoc）或应用程序类（CxxApp）中，不要放在视图类中。<br /><br />例：<br />//退出程序<br />BOOL CEditDoc::CanCloseFrame(CFrameWnd* pFrame) <br />{<br />     CFile file;<br />     if(b_Flag)     //b_Flag为文档修改标志，在修改文档时将其置为True<br />     {<br />         int t;<br />         t=::MessageBox(NULL,"文字已经改变，要存盘吗？","警告",<br />                 MB_YESNOCANCEL | MB_ICONWARNING);     //弹出提示对话框<br />         if(t==0 || t==IDCANCEL)<br />             return false;<br />         if(t==IDYES)<br />         {<br />             CString sFilter="Text File(*.txt)|*.txt||";<br />             CFileDialog m_Dlg(FALSE,"txt",NULL,OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,(LPCTSTR)sFilter,NULL);     //定制文件对话框<br /><br />             int k=m_Dlg.DoModal();     //弹出文件对话框<br />             if(k==IDCANCEL || k==0)<br />                 return false;<br />             m_PathName=m_Dlg.GetPathName();     //获取选择的文件路径名<br />            <br />             file.Open(m_PathName,CFile::modeCreate | CFile::modeWrite);<br />             file.Write(m_Text,m_TextLen);     //数据写入文件<br />             file.Close();<br />         }<br />     }<br />     return CDocument::CanCloseFrame(pFrame);<br />}</wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></wbr></p>
<img src ="http://www.cppblog.com/cs569652894/aggbug/145783.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/cs569652894/" target="_blank">CrackerCS</a> 2011-05-06 00:41 <a href="http://www.cppblog.com/cs569652894/archive/2011/05/06/145783.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item><item><title>不包含任何头文件弹出MessagBox</title><link>http://www.cppblog.com/cs569652894/archive/2011/04/14/144216.html</link><dc:creator>CrackerCS</dc:creator><author>CrackerCS</author><pubDate>Thu, 14 Apr 2011 07:07:00 GMT</pubDate><guid>http://www.cppblog.com/cs569652894/archive/2011/04/14/144216.html</guid><wfw:comment>http://www.cppblog.com/cs569652894/comments/144216.html</wfw:comment><comments>http://www.cppblog.com/cs569652894/archive/2011/04/14/144216.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.cppblog.com/cs569652894/comments/commentRss/144216.html</wfw:commentRss><trackback:ping>http://www.cppblog.com/cs569652894/services/trackbacks/144216.html</trackback:ping><description><![CDATA[<p>首先创建一个Win32 Application 工程，然后就是敲入如下代码段：<br></p>
<div style="BORDER-RIGHT: #cccccc 1px solid; PADDING-RIGHT: 5px; BORDER-TOP: #cccccc 1px solid; PADDING-LEFT: 4px; FONT-SIZE: 12pt; PADDING-BOTTOM: 4px; BORDER-LEFT: #cccccc 1px solid; WIDTH: 98%; WORD-BREAK: break-all; PADDING-TOP: 4px; BORDER-BOTTOM: #cccccc 1px solid; BACKGROUND-COLOR: #eeeeee">&nbsp;1<img id=Codehighlighter1_0_60_Open_Image onclick="this.style.display='none'; Codehighlighter1_0_60_Open_Text.style.display='none'; Codehighlighter1_0_60_Closed_Image.style.display='inline'; Codehighlighter1_0_60_Closed_Text.style.display='inline';" src="http://www.cppblog.com/Images/OutliningIndicators/ExpandedBlockStart.gif" align=top><img id=Codehighlighter1_0_60_Closed_Image onclick="this.style.display='none'; Codehighlighter1_0_60_Closed_Text.style.display='none'; Codehighlighter1_0_60_Open_Image.style.display='inline'; Codehighlighter1_0_60_Open_Text.style.display='inline';" src="http://www.cppblog.com/Images/OutliningIndicators/ContractedBlock.gif" align=top>/**//*<br>&nbsp;2<img src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif" align=top>&nbsp;HINSTANCE__*&nbsp;lpHinstance&nbsp;=&nbsp;(HWND__*)lpHWnd;<br>&nbsp;3<img src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif" align=top>&nbsp;//编译器检查类型<br>&nbsp;4<img src="http://www.cppblog.com/Images/OutliningIndicators/ExpandedBlockEnd.gif" align=top>*/<br>&nbsp;5<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top>#define&nbsp;DECLARE_HANDLE(name)&nbsp;struct&nbsp;name##__&nbsp;{&nbsp;&nbsp;int&nbsp;unused;&nbsp;&nbsp;};&nbsp;typedef&nbsp;struct&nbsp;name##__&nbsp;*name<br>&nbsp;6<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top><br>&nbsp;7<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top>DECLARE_HANDLE(HINSTANCE);<br>&nbsp;8<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top><br>&nbsp;9<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top>DECLARE_HANDLE(HWND);<br>10<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top><br>11<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top>typedef&nbsp;char*&nbsp;LPSTR;<br>12<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top><br>13<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top>typedef&nbsp;const&nbsp;char*&nbsp;LPCSTR;<br>14<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top><br>15<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top>typedef&nbsp;unsigned&nbsp;int&nbsp;UINT;<br>16<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top><br>17<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top>#define&nbsp;APIENTRY&nbsp;__stdcall<br>18<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top><br>19<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top>#define&nbsp;MB_OK&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;0x00000000L<br>20<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top><br>21<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top>#define&nbsp;NULL&nbsp;0<br>22<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top><br>23<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top>extern&nbsp;"C"&nbsp;<br>24<img id=Codehighlighter1_392_630_Open_Image onclick="this.style.display='none'; Codehighlighter1_392_630_Open_Text.style.display='none'; Codehighlighter1_392_630_Closed_Image.style.display='inline'; Codehighlighter1_392_630_Closed_Text.style.display='inline';" src="http://www.cppblog.com/Images/OutliningIndicators/ExpandedBlockStart.gif" align=top><img id=Codehighlighter1_392_630_Closed_Image onclick="this.style.display='none'; Codehighlighter1_392_630_Closed_Text.style.display='none'; Codehighlighter1_392_630_Open_Image.style.display='inline'; Codehighlighter1_392_630_Open_Text.style.display='inline';" src="http://www.cppblog.com/Images/OutliningIndicators/ContractedBlock.gif" align=top><img src="http://www.cppblog.com/Images/dot.gif">{<br>25<img src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif" align=top>&nbsp;&nbsp;int&nbsp;__stdcall&nbsp;MessageBoxA(HWND&nbsp;hWnd&nbsp;,<br>26<img src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif" align=top>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;LPCSTR&nbsp;lpText,<br>27<img src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif" align=top>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;LPCSTR&nbsp;lpCaption,<br>28<img src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif" align=top>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;UINT&nbsp;uType);<br>29<img src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif" align=top><br>30<img src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif" align=top>&nbsp;&nbsp;HWND&nbsp;__stdcall&nbsp;FindWindowA(<br>31<img src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif" align=top>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;LPCSTR&nbsp;lpClassName&nbsp;,<br>32<img src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif" align=top>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;LPCSTR&nbsp;lpWindowName);<br>33<img src="http://www.cppblog.com/Images/OutliningIndicators/ExpandedBlockEnd.gif" align=top>}<br>34<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top><br>35<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top>int&nbsp;APIENTRY&nbsp;WinMain(HINSTANCE&nbsp;hInstance,<br>36<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;HINSTANCE&nbsp;hPrevInstance,<br>37<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;LPSTR&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;lpCmdLine,<br>38<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;int&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;nCmdShow)<br>39<img id=Codehighlighter1_804_869_Open_Image onclick="this.style.display='none'; Codehighlighter1_804_869_Open_Text.style.display='none'; Codehighlighter1_804_869_Closed_Image.style.display='inline'; Codehighlighter1_804_869_Closed_Text.style.display='inline';" src="http://www.cppblog.com/Images/OutliningIndicators/ExpandedBlockStart.gif" align=top><img id=Codehighlighter1_804_869_Closed_Image onclick="this.style.display='none'; Codehighlighter1_804_869_Closed_Text.style.display='none'; Codehighlighter1_804_869_Open_Image.style.display='inline'; Codehighlighter1_804_869_Open_Text.style.display='inline';" src="http://www.cppblog.com/Images/OutliningIndicators/ContractedBlock.gif" align=top><img src="http://www.cppblog.com/Images/dot.gif">{<br>40<img src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif" align=top>&nbsp;&nbsp;MessageBoxA(NULL,"Hello",(LPCSTR)"Title",MB_OK);<br>41<img src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif" align=top><br>42<img src="http://www.cppblog.com/Images/OutliningIndicators/InBlock.gif" align=top>&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;0;<br>43<img src="http://www.cppblog.com/Images/OutliningIndicators/ExpandedBlockEnd.gif" align=top>}<br>44<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top><br>45<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top><br>46<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top><br>47<img src="http://www.cppblog.com/Images/OutliningIndicators/None.gif" align=top></div>
<img src ="http://www.cppblog.com/cs569652894/aggbug/144216.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.cppblog.com/cs569652894/" target="_blank">CrackerCS</a> 2011-04-14 15:07 <a href="http://www.cppblog.com/cs569652894/archive/2011/04/14/144216.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item></channel></rss>