关于怎么在wine中安装QQ,网上教程很多,不过能搜索到的不少都有点老了,以前用Debian5.0/6.0的时候就是按网上教程并自己实践的。用wine1.2系列安装QQ是非常折腾的,需要安装很多支撑库,而且非常容易崩溃。在wine1.3版本后,相对就容易许多了。
以现在Ubuntu11.10为例,使用的是wine1.3.28版本,安装QQ2010的步骤:
1、sudo apt-get install cups-bsd gnome-exe-thumbnailer wine1.3 ttf-droid ttf-liberation ttf-unfonts-core wine1.3-gecko winetricks
2、winetricks ie6 vcrun6 riched20
3、wine QQ2010.exe
(更新:如果需要安装QQ2011,则需要使用ppa上面的wine1.3.35版本之后的wine,但本人未测试过。)
根据状态切换原理,首先想到的是GetLastInputInfo这个函数,写了个测试程序,发现这个函数在wine里面运行是正常的。不过在gnome-terminal下启动wine /path_to_qq/qq.exe,发现重复如下信息:
fixme:crypt:SystemFunction041 (0x3d0b728, e78, 1): stub [RtlDecryptMemory]
fixme:crypt:SystemFunction040 (0x5558fc8, e78, 1): stub [RtlEncryptMemory]
fixme:winstation:OpenInputDesktop (0,0,400001cf): stub
…
使得这个函数返回true就可以解决问题了。本人采用把OpenInputDesktop之后的那个jnz换成jmp的方法来完成的。使用IDA的HexView,找到jnz对应的机器代码75 xx,把随后的十来个数字一起,作为特征码;用WinHex打开common.dll,使用上面“特征码“定位,并把75改成EB(即jnz->jmp),并把common.dll的头部checksum值清空,保存文件。再次启qq,状态切换问题解决。(缺点是:在Linux真的处于锁屏状态时,QQ仍显示在线。
在Debian下,Ctrl+Alt+d(显示桌面)后,双击托盘的qq图标,qq主界面不显示。本人解决办法是使用ShowWindow,代码如下:
// showQQ.cpp : Defines the entry point for the application. #include <windows.h> BOOL CALLBACK EnumWndProc(HWND hWnd,LPARAM lParam) { TCHAR winTitle[MAX_PATH] = {0}; ::GetWindowText(hWnd, winTitle, sizeof(winTitle)); if( 0 == strcmp(winTitle, "QQ International") || 0 == strcmp(winTitle, "QQ2011") || 0 == strcmp(winTitle, "QQ2010") // 请自行添加其它需要的QQ主窗口名 ) { ::SendMessage(hWnd, WM_SHOWWINDOW, 0, 0); ::SetWindowPos(hWnd, HWND_TOP, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE | SWP_SHOWWINDOW); ::ShowWindow(hWnd, SW_SHOWNORMAL); ::SendMessage(hWnd, WM_SHOWWINDOW, 1, 0); } return TRUE; } int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) { ::EnumWindows(EnumWndProc, (LPARAM)0); return 0; }
结语:
qq是身边人用得比较多的联系工具,但腾讯对非windows用户照顾不周,使用QQ非常艰难。希望有更好的开放协议的通讯工具,让大家有更多的选择。测试 GetLastInputInfo代码
#define _WIN32_WINNT 0x500 #include <windows.h> #include <winuser.h> #include "stdio.h" #include "time.h" int main(int argc, char* argv[]) { LASTINPUTINFO info; info.cbSize = sizeof(info); while(1) { if(GetLastInputInfo(&info)) { printf ( "%ud\n", info.dwTime); } else { printf("get last input info fail.\n"); } Sleep(1000); } return 0; }