WxMgr.cpp 5.5 KB

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