WxMgr.cpp 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. #include "stdafx.h"
  2. #include "WxMgr.h"
  3. #include "Global.h"
  4. #include "resource.h"
  5. TCHAR CWxMgr::m_szWeChatPath[MAX_PATH] = { 0 };
  6. CWxMgr::CWxMgr(CWnd *pMainWnd) :m_pMainWnd(pMainWnd)
  7. {
  8. InitButtonIds();
  9. GetWeChatPath();
  10. }
  11. CWxMgr::~CWxMgr()
  12. {
  13. }
  14. CRect CWxMgr::CalcRect()
  15. {
  16. CRect rc;
  17. int nCount = m_mapWx.size();
  18. // 高固定大小;
  19. rc.left = m_rcWxList.left + 2;
  20. rc.top = 100 * nCount + m_rcWxList.top + 20;
  21. rc.right = m_rcWxList.right - 2;
  22. rc.bottom = rc.top + 100;
  23. return rc;
  24. }
  25. // 获取空闲的id;
  26. DWORD CWxMgr::GetFreeButtonId(DWORD dwProcId)
  27. {
  28. for (pair<DWORD, DWORD> ids : m_mapIds)
  29. {
  30. if ( ids.second == 0 )
  31. {
  32. ids.second = dwProcId;
  33. return ids.first;
  34. }
  35. }
  36. return -1;
  37. }
  38. VOID CWxMgr::ReSetButtonKey(DWORD dwProcId)
  39. {
  40. for (pair<DWORD, DWORD> ids : m_mapIds)
  41. {
  42. if (ids.second == dwProcId)
  43. {
  44. ids.second = 0;
  45. break;
  46. }
  47. }
  48. }
  49. void CWxMgr::InitButtonIds()
  50. {
  51. DWORD dwStartId = 10000;
  52. for ( int i = 0; i < 10; i++)
  53. {
  54. m_mapIds.insert(pair<DWORD, DWORD>(dwStartId + i, 0));
  55. }
  56. }
  57. BOOL CWxMgr::AttachUnMgrWxProc(CWnd *pWnd)
  58. {
  59. vector<DWORD> vtWxProcId = FindAllProcess(WECHAT);
  60. for (DWORD pid : vtWxProcId)
  61. {
  62. if (m_mapWx.count(pid) == 0)
  63. {
  64. AddMapItem(pid, NULL);
  65. }
  66. }
  67. return TRUE;
  68. }
  69. BOOL CWxMgr::AttachWx()
  70. {
  71. return 0;
  72. }
  73. VOID CWxMgr::AddMapItem(DWORD dwProcId, LPCTSTR lpItemName)
  74. {
  75. WXOBJINFO wxObjInfo;
  76. // 创建微信对象;
  77. wxObjInfo.pWxObject = new CWxObject(dwProcId, g_szDynamicLibraryPath);
  78. // 创建按钮;//id使用自建的;
  79. wxObjInfo.pWxButton = new CButtonST();
  80. wxObjInfo.pWxButton->Create(lpItemName, WS_TABSTOP | WS_VISIBLE | WS_CHILD | BS_AUTOCHECKBOX, CalcRect(), m_pMainWnd, GetFreeButtonId(dwProcId));
  81. wxObjInfo.pWxButton->OffsetColor(CButtonST::BTNST_COLOR_BK_IN, 60);
  82. //wxObjInfo.pWxButton->DrawTransparent();
  83. wxObjInfo.pWxButton->SetIcon(ICON_CHECK);
  84. wxObjInfo.pWxButton->ShowWindow(SW_SHOW);
  85. // 创建聊天窗口;//Id使用进程id;
  86. wxObjInfo.pWxChatWnd = new CStatic();
  87. wxObjInfo.pWxChatWnd->Create(lpItemName, WS_CHILD | WS_VISIBLE | WS_TABSTOP | SS_GRAYRECT, m_rcWxShow, m_pMainWnd, dwProcId);
  88. wxObjInfo.pWxChatWnd->ShowWindow(SW_SHOW);
  89. // 附加到窗口;
  90. wxObjInfo.pWxObject->InjectDynamicLibrary();
  91. if (wxObjInfo.pWxObject->FindWxMainWnd())
  92. {
  93. wxObjInfo.pWxObject->Attach2MainWnd(wxObjInfo.pWxChatWnd, FALSE);
  94. }
  95. else if (wxObjInfo.pWxObject->FindWxLoginWnd())
  96. {
  97. wxObjInfo.pWxObject->Attach2MainWnd(wxObjInfo.pWxChatWnd, TRUE);
  98. }
  99. m_mapWx.insert(pair<DWORD, WXOBJINFO>(dwProcId, wxObjInfo));
  100. }
  101. VOID CWxMgr::RemoveMapItem(DWORD dwProcId)
  102. {
  103. for (pair<DWORD, WXOBJINFO> wxinfo : m_mapWx)
  104. {
  105. if (wxinfo.first == dwProcId)
  106. {
  107. wxinfo.second.pWxObject->DetachWxWnd();
  108. delete wxinfo.second.pWxObject;
  109. delete wxinfo.second.pWxButton;
  110. delete wxinfo.second.pWxChatWnd;
  111. // 重转按钮关联的进程id;
  112. ReSetButtonKey(dwProcId);
  113. // 下方的WxButton向上移动;
  114. // ...
  115. break;
  116. }
  117. }
  118. // 移除key;
  119. m_mapWx.erase(dwProcId);
  120. }
  121. VOID CWxMgr::DetachAllWxProc()
  122. {
  123. for (pair<DWORD, WXOBJINFO> wxinfo : m_mapWx)
  124. {
  125. wxinfo.second.pWxObject->DetachWxWnd();
  126. delete wxinfo.second.pWxObject;
  127. delete wxinfo.second.pWxButton;
  128. delete wxinfo.second.pWxChatWnd;
  129. // 重转按钮关联的进程id;
  130. ReSetButtonKey(wxinfo.first);
  131. // 下方的WxButton向上移动;
  132. // ...
  133. }
  134. m_mapWx.clear();
  135. }
  136. BOOL CWxMgr::OpenWeChat()
  137. {
  138. // 关闭互斥句柄;
  139. PatchWeChat();
  140. STARTUPINFO si;
  141. PROCESS_INFORMATION pi;
  142. ZeroMemory(&si, sizeof(si));
  143. si.cb = sizeof(si);
  144. ZeroMemory(&pi, sizeof(pi));
  145. si.dwFlags = STARTF_USESHOWWINDOW; // 指定wShowWindow成员有效
  146. si.wShowWindow = SW_HIDE; // 设置创建进程时,窗口不显示,
  147. // 为FALSE的话则不显示
  148. BOOL bRet = ::CreateProcess(
  149. m_szWeChatPath, // 不在此指定可执行文件的文件名
  150. NULL, // 命令行参数
  151. NULL, // 默认进程安全性
  152. NULL, // 默认线程安全性
  153. FALSE, // 指定当前进程内的句柄不可以被子进程继承
  154. NULL,
  155. NULL, // 使用本进程的环境变量
  156. NULL, // 使用本进程的驱动器和目录
  157. &si,
  158. &pi);
  159. if (bRet)
  160. {
  161. //WaitForSingleObject(pi.hProcess, INFINITE);
  162. //WaitForInputIdle(pi.hProcess, INFINITE);
  163. //Sleep(1500); // 启动微信需要1秒左右;
  164. // 既然我们不使用两个句柄,最好是立刻将它们关闭
  165. ::CloseHandle(pi.hThread);
  166. ::CloseHandle(pi.hProcess);
  167. // 当进程挂起时,是无法修改关闭微信句柄;
  168. AddMapItem(pi.dwProcessId, NULL);
  169. }
  170. return 0;
  171. }
  172. BOOL CWxMgr::GetWeChatPath()
  173. {
  174. static BOOL bGotPath = FALSE;
  175. if (!bGotPath)
  176. {
  177. // 通过注册表获取微信安装目录;
  178. HKEY hKey = NULL;
  179. if (ERROR_SUCCESS != RegOpenKey(HKEY_CURRENT_USER, _T("Software\\Tencent\\WeChat"), &hKey))
  180. {
  181. return FALSE;
  182. }
  183. DWORD Type = REG_SZ;
  184. DWORD cbData = MAX_PATH * sizeof(WCHAR);
  185. if (ERROR_SUCCESS != RegQueryValueEx(hKey, _T("InstallPath"), 0, &Type, (LPBYTE)m_szWeChatPath, &cbData))
  186. {
  187. RegCloseKey(hKey);
  188. return FALSE;
  189. }
  190. PathAppend(m_szWeChatPath, _T("WeChat.exe"));
  191. bGotPath = TRUE;
  192. }
  193. return bGotPath;
  194. }