WxMgr.cpp 5.5 KB

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