mypingdlg.cpp

来自「Visual_C++.NET实用编程百例」· C++ 代码 · 共 316 行

CPP
316
字号
// myPingDlg.cpp : 实现文件
//

#include "stdafx.h"
#include "myPing.h"
#include "myPingDlg.h"
#include ".\mypingdlg.h"
#include "afxsock.h"

#ifdef _DEBUG
#define new DEBUG_NEW
#endif


// 用于应用程序“关于”菜单项的 CAboutDlg 对话框

class CAboutDlg : public CDialog
{
public:
	CAboutDlg();

// 对话框数据
	enum { IDD = IDD_ABOUTBOX };

	protected:
	virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持

// 实现
protected:
	DECLARE_MESSAGE_MAP()
};

CAboutDlg::CAboutDlg() : CDialog(CAboutDlg::IDD)
{
}

void CAboutDlg::DoDataExchange(CDataExchange* pDX)
{
	CDialog::DoDataExchange(pDX);
}

BEGIN_MESSAGE_MAP(CAboutDlg, CDialog)
END_MESSAGE_MAP()


// CmyPingDlg 对话框



CmyPingDlg::CmyPingDlg(CWnd* pParent /*=NULL*/)
	: CDialog(CmyPingDlg::IDD, pParent)
	, m_timeOut(0)
	, m_packetSize(0)
{
	m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}

void CmyPingDlg::DoDataExchange(CDataExchange* pDX)
{
	CDialog::DoDataExchange(pDX);
	DDX_Control(pDX, IDC_LIST2, m_listBox);
	DDX_Text(pDX, IDC_EDIT1, m_timeOut);
	DDX_Text(pDX, IDC_EDIT2, m_packetSize);
	DDX_Control(pDX, IDC_IPADDRESS1, m_IP);
}

BEGIN_MESSAGE_MAP(CmyPingDlg, CDialog)
	ON_WM_SYSCOMMAND()
	ON_WM_PAINT()
	ON_WM_QUERYDRAGICON()
	//}}AFX_MSG_MAP
	ON_WM_TIMER()
	ON_BN_CLICKED(IDOK, OnBnClickedOk)
END_MESSAGE_MAP()


// CmyPingDlg 消息处理程序

BOOL CmyPingDlg::OnInitDialog()
{
	CDialog::OnInitDialog();
	m_sendBuf=NULL;
	m_recBuf=NULL;

	// 将\“关于...\”菜单项添加到系统菜单中。

	// IDM_ABOUTBOX 必须在系统命令范围内。
	ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
	ASSERT(IDM_ABOUTBOX < 0xF000);

	CMenu* pSysMenu = GetSystemMenu(FALSE);
	if (pSysMenu != NULL)
	{
		CString strAboutMenu;
		strAboutMenu.LoadString(IDS_ABOUTBOX);
		if (!strAboutMenu.IsEmpty())
		{
			pSysMenu->AppendMenu(MF_SEPARATOR);
			pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
		}
	}

	// 设置此对话框的图标。当应用程序主窗口不是对话框时,框架将自动
	//  执行此操作
	SetIcon(m_hIcon, TRUE);			// 设置大图标
	SetIcon(m_hIcon, FALSE);		// 设置小图标

	// TODO: 在此添加额外的初始化代码
	
	return TRUE;  // 除非设置了控件的焦点,否则返回 TRUE
}

void CmyPingDlg::OnSysCommand(UINT nID, LPARAM lParam)
{
	if ((nID & 0xFFF0) == IDM_ABOUTBOX)
	{
		CAboutDlg dlgAbout;
		dlgAbout.DoModal();
	}
	else
	{
		CDialog::OnSysCommand(nID, lParam);
	}
}

// 如果向对话框添加最小化按钮,则需要下面的代码
//  来绘制该图标。对于使用文档/视图模型的 MFC 应用程序,
//  这将由框架自动完成。

void CmyPingDlg::OnPaint() 
{
	if (IsIconic())
	{
		CPaintDC dc(this); // 用于绘制的设备上下文

		SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), 0);

		// 使图标在工作矩形中居中
		int cxIcon = GetSystemMetrics(SM_CXICON);
		int cyIcon = GetSystemMetrics(SM_CYICON);
		CRect rect;
		GetClientRect(&rect);
		int x = (rect.Width() - cxIcon + 1) / 2;
		int y = (rect.Height() - cyIcon + 1) / 2;

		// 绘制图标
		dc.DrawIcon(x, y, m_hIcon);
	}
	else
	{
		CDialog::OnPaint();
	}
}

//当用户拖动最小化窗口时系统调用此函数取得光标显示。
HCURSOR CmyPingDlg::OnQueryDragIcon()
{
	return static_cast<HCURSOR>(m_hIcon);
}
USHORT CmyPingDlg::GetSum(USHORT *buf,int size)
{
	unsigned long ret=0;
	while(size>1)
	{
		ret+=*buf++;
		size-=sizeof(USHORT);
	}
	if(size)
		ret+=*(UCHAR*)buf;
	ret=(ret>>16)+(ret&0xffff);
	ret+=(ret>>16);
	return (USHORT)(~ret);
}

void CmyPingDlg::GetICMPPacket()
{
	char *real_send;
	ICMP_HEADER *pHead;
	pHead=(ICMP_HEADER*)m_sendBuf;
	pHead->type =ICMP_ECHO_TYPE;
	pHead->code =0;
	pHead->ID =GetCurrentProcessId();
	real_send=m_sendBuf+sizeof(ICMP_HEADER);
	memset(real_send,'z',m_dataSize-sizeof(ICMP_HEADER));
}
BOOL CmyPingDlg::AnalyseBuf()
{
	IP_HEADER *pIPHead;
	ICMP_HEADER *pICMPHead;
	pIPHead=(IP_HEADER*)m_recBuf;
	CString m_str;
	int len=pIPHead->len <<2;
	if((len+ICMP_MIN)>nRec)
	{
		m_strErr.Format (IDP_SOCKETS_ECHODATA_FAILED);
		ShowMsg(m_strErr);
		return FALSE;
	}
	pICMPHead=(ICMP_HEADER *)(m_recBuf+len);
	if(pICMPHead->type !=0)
	{
		m_strErr.Format (IDP_SOCKETS_ECHOTYPE_FAILED);
		ShowMsg(m_strErr);
		return FALSE;
	}
	if((USHORT)GetCurrentProcessId()!=pICMPHead->ID)
	{
		m_strErr.LoadString (IDP_SOCKETS_MISMATCH);
		ShowMsg(m_strErr);
		return FALSE;
	}
	m_str.Format (IDP_SOCKETS_PING_INFO,inet_ntoa(m_addrFrom.sin_addr ),nRec,GetTickCount()-((ICMP_HEADER*)m_sendBuf)->timestamp ,((IP_HEADER*)m_recBuf)->ttl );
	ShowMsg(m_str);
	return TRUE;
}

void CmyPingDlg::ShowMsg(LPCSTR lpsz)
{
	m_listBox.AddString(lpsz);
}
void CmyPingDlg::OnTimer(UINT nIDEvent)
{
	// TODO: 在此添加消息处理程序代码和/或调用默认值
	CDialog::OnTimer(nIDEvent);
	if(nPing_Count==MAX_PING_COUNT)
	{
		KillTimer(m_timer);
		closesocket(m_sock);
		return;
	}
	nPing_Count++;
	((ICMP_HEADER*)m_sendBuf)->cksum=0;
	((ICMP_HEADER*)m_sendBuf)->timestamp=GetTickCount();
	((ICMP_HEADER*)m_sendBuf)->cksum=GetSum((USHORT*)m_sendBuf,m_dataSize);
	if((nSend=sendto(m_sock,m_sendBuf,m_dataSize,0,(struct sockaddr*)&m_addrDest,sizeof(m_addrDest)))==SOCKET_ERROR)
	{
		KillTimer(m_timer);
		m_strErr.Format (IDP_SOCKETS_SEND_FAILED,WSAGetLastError());
		ShowMsg(m_strErr);
		return;
	}
	int nAddrFrom =sizeof(m_addrFrom);
	if((nRec=recvfrom(m_sock,m_recBuf,m_dataSize+MAX_IPHEADER,0,(struct sockaddr*)&m_addrFrom,&nAddrFrom))==SOCKET_ERROR)
	{
		KillTimer(m_timer);
		if(WSAGetLastError()==WSAETIMEDOUT)
		{
			m_strErr.Format (IDP_SOCKETS_TIMEOUT);
			ShowMsg(m_strErr);
			return;
		}
		else
		{
			m_strErr.Format (IDP_SOCKETS_REC_FAILED);
			ShowMsg(m_strErr);
			return;
		}
	}
	AnalyseBuf();
}
BOOL CmyPingDlg::StartPing ()
{
	nPing_Count=0;
	if(m_sendBuf!=NULL)
	{
		delete m_sendBuf;
		m_sendBuf=NULL;
	}
	if(m_recBuf!=NULL)
	{
		delete m_recBuf;
		m_recBuf=NULL;
	}
	if(m_sock!=INVALID_SOCKET)
	{
		closesocket(m_sock);
		m_sock=INVALID_SOCKET;
	}
	if((m_sock=socket(AF_INET,SOCK_RAW,IPPROTO_ICMP))==INVALID_SOCKET)
	{
		m_strErr.Format (IDP_SOCKETS_CREAT_FAILED,WSAGetLastError());
		ShowMsg(m_strErr);
		return FALSE;
	}
	if((setsockopt(m_sock,SOL_SOCKET,SO_RCVTIMEO,(char*)&m_timeOut,sizeof(m_timeOut)))==SOCKET_ERROR)
	{
		m_strErr.Format (IDP_SOCKETS_SETOPT_FAILED,WSAGetLastError());
		ShowMsg(m_strErr);
		return FALSE;
	}
	m_addrDest.sin_family =AF_INET;
	m_addrDest.sin_addr.s_addr=inet_addr(m_hostAddr);
	m_dataSize=m_packetSize+sizeof(ICMP_HEADER);
	m_sendBuf=new char[m_dataSize];
	m_recBuf=new char[m_dataSize+MAX_IPHEADER];
	GetICMPPacket();
	m_timer=SetTimer(1,500,NULL);
	return TRUE;
}
void CmyPingDlg::OnBnClickedOk()
{
	m_listBox.ResetContent ();
	BYTE m_addrField[4];
	UpdateData();
	DWORD m_test;
	m_IP.GetAddress (m_test);
	if(m_timeOut==0||m_packetSize==0||m_test==0)
	{
		ShowMsg("请重新填写数据");
		return;
	}
	m_IP.GetAddress (m_addrField[0],m_addrField[1],m_addrField[2],m_addrField[3]);
	m_hostAddr.Format ("%d.%d.%d.%d",m_addrField[0],m_addrField[1],m_addrField[2],m_addrField[3]);
	StartPing();
}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?