ClientDlg.cpp 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312
  1. // ClientDlg.cpp : implementation file
  2. //
  3. #include "stdafx.h"
  4. #include "Client.h"
  5. #include "ClientDlg.h"
  6. #include "afxdialogex.h"
  7. // CClientDlg dialog
  8. #define DEFAULT_CONTENT _T("text to be sent")
  9. #define DEFAULT_ADDRESS _T("127.0.0.1")
  10. #define DEFAULT_PORT _T("5555")
  11. CClientDlg* CClientDlg::m_spThis = nullptr;
  12. CClientDlg::CClientDlg(CWnd* pParent /*=NULL*/)
  13. : CDialogEx(CClientDlg::IDD, pParent)
  14. {
  15. m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
  16. m_spThis = this;
  17. // 初始化 SSL 全局环境参数
  18. ::HP_SSL_Initialize(SSL_SM_CLIENT, g_s_iVerifyMode, g_s_lpszPemCertFile, g_s_lpszPemKeyFile, g_s_lpszKeyPasswod, g_s_lpszCAPemCertFileOrPath);
  19. VERIFY(::HP_SSL_IsValid());
  20. // 创建监听器对象
  21. m_pListener = ::Create_HP_TcpPullClientListener();
  22. // 创建 Socket 对象
  23. m_pClient = ::Create_HP_SSLPullClient(m_pListener);
  24. // 设置 Socket 监听器回调函数
  25. ::HP_Set_FN_Client_OnConnect(m_pListener, OnConnect);
  26. ::HP_Set_FN_Client_OnHandShake(m_pListener, OnHandShake);
  27. ::HP_Set_FN_Client_OnSend(m_pListener, OnSend);
  28. ::HP_Set_FN_Client_OnPullReceive(m_pListener, OnReceive);
  29. ::HP_Set_FN_Client_OnClose(m_pListener, OnClose);
  30. }
  31. CClientDlg::~CClientDlg()
  32. {
  33. // 销毁 Socket 对象
  34. ::Destroy_HP_SSLPullClient(m_pClient);
  35. // 销毁监听器对象
  36. ::Destroy_HP_TcpPullClientListener(m_pListener);
  37. // 清理 SSL 全局运行环境
  38. ::HP_SSL_Cleanup();
  39. }
  40. void CClientDlg::DoDataExchange(CDataExchange* pDX)
  41. {
  42. CDialogEx::DoDataExchange(pDX);
  43. DDX_Control(pDX, IDC_CONTENT, m_Content);
  44. DDX_Control(pDX, IDC_SEND, m_Send);
  45. DDX_Control(pDX, IDC_INFO, m_Info);
  46. DDX_Control(pDX, IDC_ADDRESS, m_Address);
  47. DDX_Control(pDX, IDC_PORT, m_Port);
  48. DDX_Control(pDX, IDC_ASYNC, m_Async);
  49. DDX_Control(pDX, IDC_START, m_Start);
  50. DDX_Control(pDX, IDC_STOP, m_Stop);
  51. }
  52. BEGIN_MESSAGE_MAP(CClientDlg, CDialogEx)
  53. ON_WM_PAINT()
  54. ON_WM_QUERYDRAGICON()
  55. ON_BN_CLICKED(IDC_SEND, &CClientDlg::OnBnClickedSend)
  56. ON_BN_CLICKED(IDC_START, &CClientDlg::OnBnClickedStart)
  57. ON_BN_CLICKED(IDC_STOP, &CClientDlg::OnBnClickedStop)
  58. ON_MESSAGE(USER_INFO_MSG, OnUserInfoMsg)
  59. ON_WM_VKEYTOITEM()
  60. END_MESSAGE_MAP()
  61. // CClientDlg message handlers
  62. BOOL CClientDlg::OnInitDialog()
  63. {
  64. CDialogEx::OnInitDialog();
  65. // Set the icon for this dialog. The framework does this automatically
  66. // when the application's main window is not a dialog
  67. SetIcon(m_hIcon, TRUE); // Set big icon
  68. SetIcon(m_hIcon, FALSE); // Set small icon
  69. // TODO: Add extra initialization here
  70. m_Content.SetWindowText(DEFAULT_CONTENT);
  71. m_Address.SetWindowText(DEFAULT_ADDRESS);
  72. m_Port.SetWindowText(DEFAULT_PORT);
  73. m_Async.SetCheck(BST_CHECKED);
  74. ::SetMainWnd(this);
  75. ::SetInfoList(&m_Info);
  76. SetAppState(ST_STOPPED);
  77. m_bAsyncConn = FALSE;
  78. return TRUE; // return TRUE unless you set the focus to a control
  79. }
  80. // If you add a minimize button to your dialog, you will need the code below
  81. // to draw the icon. For MFC applications using the document/view model,
  82. // this is automatically done for you by the framework.
  83. void CClientDlg::OnPaint()
  84. {
  85. if (IsIconic())
  86. {
  87. CPaintDC dc(this); // device context for painting
  88. SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), 0);
  89. // Center icon in client rectangle
  90. int cxIcon = GetSystemMetrics(SM_CXICON);
  91. int cyIcon = GetSystemMetrics(SM_CYICON);
  92. CRect rect;
  93. GetClientRect(&rect);
  94. int x = (rect.Width() - cxIcon + 1) / 2;
  95. int y = (rect.Height() - cyIcon + 1) / 2;
  96. // Draw the icon
  97. dc.DrawIcon(x, y, m_hIcon);
  98. }
  99. else
  100. {
  101. CDialogEx::OnPaint();
  102. }
  103. }
  104. // The system calls this function to obtain the cursor to display while the user drags
  105. // the minimized window.
  106. HCURSOR CClientDlg::OnQueryDragIcon()
  107. {
  108. return static_cast<HCURSOR>(m_hIcon);
  109. }
  110. BOOL CClientDlg::PreTranslateMessage(MSG* pMsg)
  111. {
  112. if (
  113. pMsg->message == WM_KEYDOWN
  114. &&( pMsg->wParam == VK_ESCAPE
  115. || pMsg->wParam == VK_CANCEL
  116. || pMsg->wParam == VK_RETURN
  117. ))
  118. return TRUE;
  119. return CDialog::PreTranslateMessage(pMsg);
  120. }
  121. void CClientDlg::SetAppState(EnAppState state)
  122. {
  123. m_enState = state;
  124. if(this->GetSafeHwnd() == nullptr)
  125. return;
  126. m_Async.EnableWindow(m_enState == ST_STOPPED);
  127. m_Start.EnableWindow(m_enState == ST_STOPPED);
  128. m_Stop.EnableWindow(m_enState == ST_STARTED);
  129. m_Send.EnableWindow(m_enState == ST_STARTED);
  130. m_Address.EnableWindow(m_enState == ST_STOPPED);
  131. m_Port.EnableWindow(m_enState == ST_STOPPED);
  132. }
  133. void CClientDlg::OnBnClickedSend()
  134. {
  135. USES_CONVERSION;
  136. static DWORD SEQ = 0;
  137. CString strContent;
  138. m_Content.GetWindowText(strContent);
  139. smart_simple_ptr<CBufferPtr> buffer = ::GeneratePkgBuffer(++SEQ, _T("伤神小怪兽"), 23, strContent);
  140. if(::HP_Client_Send(m_pClient, buffer->Ptr(), (int)buffer->Size()))
  141. ::LogSend(::HP_Client_GetConnectionID(m_pClient), strContent);
  142. else
  143. ::LogSendFail(::HP_Client_GetConnectionID(m_pClient), ::SYS_GetLastError(), ::HP_GetSocketErrorDesc(SE_DATA_SEND));
  144. }
  145. void CClientDlg::OnBnClickedStart()
  146. {
  147. SetAppState(ST_STARTING);
  148. CString strAddress;
  149. CString strPort;
  150. m_Address.GetWindowText(strAddress);
  151. m_Port.GetWindowText(strPort);
  152. USHORT usPort = (USHORT)_ttoi(strPort);
  153. m_bAsyncConn = m_Async.GetCheck();
  154. m_pkgInfo.Reset();
  155. ::LogClientStarting(strAddress, usPort);
  156. if(::HP_Client_Start(m_pClient, strAddress, usPort, m_bAsyncConn))
  157. {
  158. }
  159. else
  160. {
  161. ::LogClientStartFail(::HP_Client_GetLastError(m_pClient), HP_Client_GetLastErrorDesc(m_pClient));
  162. SetAppState(ST_STOPPED);
  163. }
  164. }
  165. void CClientDlg::OnBnClickedStop()
  166. {
  167. SetAppState(ST_STOPPING);
  168. if(::HP_Client_Stop(m_pClient))
  169. ::LogClientStopping(::HP_Client_GetConnectionID(m_pClient));
  170. else
  171. ASSERT(FALSE);
  172. }
  173. int CClientDlg::OnVKeyToItem(UINT nKey, CListBox* pListBox, UINT nIndex)
  174. {
  175. if(nKey == 'C')
  176. pListBox->ResetContent();
  177. return __super::OnVKeyToItem(nKey, pListBox, nIndex);
  178. }
  179. LRESULT CClientDlg::OnUserInfoMsg(WPARAM wp, LPARAM lp)
  180. {
  181. info_msg* msg = (info_msg*)wp;
  182. ::LogInfoMsg(msg);
  183. return 0;
  184. }
  185. En_HP_HandleResult CClientDlg::OnConnect(HP_Client pClient)
  186. {
  187. TCHAR szAddress[40];
  188. int iAddressLen = sizeof(szAddress) / sizeof(TCHAR);
  189. USHORT usPort;
  190. ::HP_Client_GetLocalAddress(pClient, szAddress, &iAddressLen, &usPort);
  191. ::PostOnConnect(::HP_Client_GetConnectionID(pClient), szAddress, usPort);
  192. return HR_OK;
  193. }
  194. En_HP_HandleResult CClientDlg::OnHandShake(HP_Client pClient)
  195. {
  196. ::PostOnHandShake(::HP_Client_GetConnectionID(pClient));
  197. m_spThis->SetAppState(ST_STARTED);
  198. return HR_OK;
  199. }
  200. En_HP_HandleResult CClientDlg::OnSend(HP_Client pClient, const BYTE* pData, int iLength)
  201. {
  202. ::PostOnSend(::HP_Client_GetConnectionID(pClient), pData, iLength);
  203. return HR_OK;
  204. }
  205. En_HP_HandleResult CClientDlg::OnReceive(HP_Client pClient, int iLength)
  206. {
  207. int required = m_spThis->m_pkgInfo.length;
  208. int remain = iLength;
  209. while(remain >= required)
  210. {
  211. remain -= required;
  212. CBufferPtr buffer(required);
  213. En_HP_FetchResult result = ::HP_TcpPullClient_Fetch(pClient, buffer, (int)buffer.Size());
  214. if(result == FR_OK)
  215. {
  216. if(m_spThis->m_pkgInfo.is_header)
  217. {
  218. TPkgHeader* pHeader = (TPkgHeader*)buffer.Ptr();
  219. TRACE("[Client] head -> seq: %d, body_len: %d\n", pHeader->seq, pHeader->body_len);
  220. required = pHeader->body_len;
  221. }
  222. else
  223. {
  224. TPkgBody* pBody = (TPkgBody*)buffer.Ptr();
  225. TRACE("[Client] body -> name: %s, age: %d, desc: %s\n", pBody->name, pBody->age, pBody->desc);
  226. required = sizeof(TPkgHeader);
  227. }
  228. m_spThis->m_pkgInfo.is_header = !m_spThis->m_pkgInfo.is_header;
  229. m_spThis->m_pkgInfo.length = required;
  230. ::PostOnReceive(::HP_Client_GetConnectionID(pClient), buffer, (int)buffer.Size());
  231. }
  232. }
  233. return HR_OK;
  234. }
  235. En_HP_HandleResult CClientDlg::OnClose(HP_Client pClient, En_HP_SocketOperation enOperation, int iErrorCode)
  236. {
  237. iErrorCode == SE_OK ? ::PostOnClose(::HP_Client_GetConnectionID(pClient)):
  238. ::PostOnError(::HP_Client_GetConnectionID(pClient), enOperation, iErrorCode);
  239. m_spThis->SetAppState(ST_STOPPED);
  240. return HR_OK;
  241. }