首页 > 代码库 > 获得QQ聊天输入框中的内容

获得QQ聊天输入框中的内容

// 首先得到输入框的句柄。通过spy++这类工具分析,聊天窗体的类名为“#32770”
// 但当前系统里不只一个类名为“#32770”的窗体,这就需要全体遍历一次。
// 类名为“#32770”标题含“聊天”基本能确定。为保险起见还判断一下所在进程是否为“qq.exe”
uses PsAPI, RichEdit;
function Process_ReadRichEditText(AHandle: THandle): WideString;
var
  vGetTextEx: GETTEXTEX;
  vGetTextLengthEx: GETTEXTLENGTHEX;
  L: Integer;
  vProcessId: DWORD;
  vProcess: THandle;
  vPointer: Pointer;
  vNumberOfBytesRead: Cardinal;
begin
  Result := ‘‘;
  if not IsWindow(AHandle) then Exit;
  GetWindowThreadProcessId(AHandle, @vProcessId);
  vProcess := OpenProcess(PROCESS_VM_OPERATION or PROCESS_VM_READ or
    PROCESS_VM_WRITE, False, vProcessId);
  try
    vPointer := VirtualAllocEx(vProcess, nil, 4096, MEM_RESERVE or MEM_COMMIT,
      PAGE_READWRITE);
    vGetTextLengthEx.flags := GTL_DEFAULT;
    vGetTextLengthEx.codepage := 1200; // Unicode
    WriteProcessMemory(vProcess, vPointer, @vGetTextLengthEx,
      SizeOf(vGetTextLengthEx), vNumberOfBytesRead);
    L := SendMessage(AHandle, EM_GETTEXTLENGTHEX, Integer(vPointer), 0);
    VirtualFreeEx(vProcess, vPointer, 0, MEM_RELEASE);
    if L <= 0 then Exit;
    vPointer := VirtualAllocEx(vProcess, nil, SizeOf(vGetTextEx) + L * 2 + 2,
      MEM_RESERVE or MEM_COMMIT, PAGE_READWRITE);
    SetLength(Result, L);
    vGetTextEx.cb := L * 2 + 2;
    vGetTextEx.flags := GT_DEFAULT;
    vGetTextEx.codepage := 1200; // Unicode
    vGetTextEx.lpDefaultChar := nil;
    vGetTextEx.lpUsedDefChar := nil;
    WriteProcessMemory(vProcess, vPointer, @vGetTextEx,
      SizeOf(vGetTextEx), vNumberOfBytesRead);
    SendMessage(AHandle, EM_GETTEXTEX, Integer(vPointer),
      Integer(vPointer) + SizeOf(vGetTextEx));
    ReadProcessMemory(vProcess, Pointer(Integer(vPointer) + SizeOf(vGetTextEx)),
      @Result[1], L * 2, vNumberOfBytesRead);
    VirtualFreeEx(vProcess, vPointer, 0, MEM_RELEASE);
  finally
    CloseHandle(vProcess);
  end;
end; { Process_ReadRichEditText }
function GetProcessName(AProcessID: THandle): string;
var
  vBuffer: array[0..MAX_PATH] of Char;
  vProcess: THandle;
begin
  vProcess := OpenProcess(PROCESS_QUERY_INFORMATION or PROCESS_VM_READ, False,
    AProcessID);
  try
    if GetModuleBaseName(vProcess, 0, vBuffer, SizeOf(vBuffer)) > 0 then
      Result := vBuffer
    else Result := ‘‘;
  finally
    CloseHandle(vProcess);
  end;
end; { GetProcessName }
function EnumChild(hwnd: HWND; lParam: LPARAM): BOOL; stdcall;
var
  vBuffer: array[0..255] of Char;
begin
  Result := True;
  if not IsWindowVisible(hwnd) then Exit; // 不可见
  GetClassName(hwnd, vBuffer, SizeOf(vBuffer));
  if SameText(vBuffer, ‘RichEdit20A‘) then
  begin
    if GetWindowLong(hwnd, GWL_STYLE) and ES_READONLY <> ES_READONLY then // 非只读
    begin
      PInteger(lParam)^ := hwnd;
      Result := False;
    end;
  end;
end; { EnumChild }
function EnumFunc(hwnd: HWND; lParam: LPARAM): BOOL; stdcall;
var
  vBuffer: array[0..255] of Char;
  vProcessId: THandle;
begin
  Result := True;
  if not IsWindowVisible(hwnd) then Exit; // 不可见
  GetClassName(hwnd, vBuffer, SizeOf(vBuffer));
  if SameText(vBuffer, ‘#32770‘) then
  begin
    GetWindowThreadProcessId(hwnd, vProcessId);
    if SameText(GetProcessName(vProcessId), ‘qq.exe‘) then
    begin
      GetWindowText(hwnd, vBuffer, SizeOf(vBuffer));
      if Pos(‘聊天中‘, vBuffer) > 0 then // 标题中含"聊天中"
      begin
        EnumChildWindows(hwnd, @EnumChild, lParam);
        Result := False;
      end;
    end;
  end;
end; { EnumFunc }
procedure TForm1.Button1Click(Sender: TObject);
var
  vHandle: THandle;
begin
  vHandle := 0;
  EnumWindows(@EnumFunc, Integer(@vHandle));
  if vHandle = 0 then Exit;
  Memo1.Text := Process_ReadRichEditText(vHandle);
end;
技术分享using System.Runtime.InteropServices;
技术分享
技术分享[DllImport("User32.DLL")]
技术分享public static extern int SendMessage(IntPtr hWnd, uint Msg, int wParam, int lParam);
技术分享
技术分享public delegate bool WNDENUMPROC(IntPtr hwnd, int lParam);
技术分享
技术分享[DllImport("user32.dll")]
技术分享public static extern int EnumWindows(WNDENUMPROC lpEnumFunc, int lParam);
技术分享[DllImport("user32.dll")]
技术分享public static extern int EnumChildWindows(IntPtr hWndParent, 
技术分享    WNDENUMPROC lpEnumFunc, int lParam);
技术分享[DllImport("user32.dll")]
技术分享public static extern int GetWindowText(IntPtr hWnd, StringBuilder lpString,
技术分享    int nMaxCount);
技术分享[DllImport("user32.dll")]
技术分享public static extern int GetClassName(IntPtr hWnd, StringBuilder lpClassName,
技术分享    int nMaxCount);
技术分享[DllImport("user32.dll")]
技术分享public static extern bool IsWindow(IntPtr hWnd);
技术分享[DllImport("user32.dll")]
技术分享public static extern bool IsWindowVisible(IntPtr hWnd);
技术分享[DllImport("user32.DLL")]
技术分享public static extern IntPtr FindWindowEx(IntPtr hwndParent,
技术分享    IntPtr hwndChildAfter, string lpszClass, string lpszWindow);
技术分享[DllImport("user32.dll")]
技术分享public static extern uint GetWindowThreadProcessId(IntPtr hWnd,
技术分享    out uint dwProcessId);
技术分享
技术分享[DllImport("psapi.dll")]
技术分享public static extern uint GetModuleBaseName(IntPtr hProcess, IntPtr hModule,
技术分享    StringBuilder lpBaseName, uint nSize);
技术分享
技术分享public const uint PROCESS_VM_OPERATION = 0x0008;
技术分享public const uint PROCESS_VM_READ = 0x0010;
技术分享public const uint PROCESS_VM_WRITE = 0x0020;
技术分享public const uint PROCESS_QUERY_INFORMATION = 0x0400;
技术分享
技术分享[DllImport("kernel32.dll")]
技术分享public static extern IntPtr OpenProcess(uint dwDesiredAccess,
技术分享    bool bInheritHandle, uint dwProcessId);
技术分享
技术分享[DllImport("kernel32.dll")]
技术分享public static extern bool CloseHandle(IntPtr handle);
技术分享
技术分享[DllImport("user32.dll")]
技术分享public static extern int GetWindowLong(IntPtr hWnd, int nIndex);
技术分享public const int GWL_STYLE = -16;
技术分享public const int ES_READONLY = 0x800;
技术分享
技术分享public const uint MEM_COMMIT = 0x1000;
技术分享public const uint MEM_RELEASE = 0x8000;
技术分享
技术分享public const uint MEM_RESERVE = 0x2000;
技术分享public const uint PAGE_READWRITE = 4;
技术分享
技术分享[DllImport("kernel32.dll")]
技术分享public static extern IntPtr VirtualAllocEx(IntPtr hProcess, IntPtr lpAddress,
技术分享    uint dwSize, uint flAllocationType, uint flProtect);
技术分享
技术分享[DllImport("kernel32.dll")]
技术分享public static extern bool VirtualFreeEx(IntPtr hProcess, IntPtr lpAddress,
技术分享   uint dwSize, uint dwFreeType);
技术分享
技术分享[DllImport("kernel32.dll")]
技术分享public static extern bool WriteProcessMemory(IntPtr hProcess, IntPtr lpBaseAddress,
技术分享   IntPtr lpBuffer, int nSize, ref uint vNumberOfBytesRead);
技术分享
技术分享[DllImport("kernel32.dll")]
技术分享public static extern bool ReadProcessMemory(IntPtr hProcess, IntPtr lpBaseAddress,
技术分享   IntPtr lpBuffer, int nSize, ref uint vNumberOfBytesRead); 
技术分享
技术分享private IntPtr richHandle;
技术分享
技术分享public string GetProcessName(uint AProcessId)
技术分享{
技术分享    StringBuilder vBuffer = new StringBuilder(256);
技术分享    IntPtr vProcess = OpenProcess(
技术分享        PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, false, AProcessId);
技术分享    try
技术分享    {
技术分享        if (GetModuleBaseName(vProcess, IntPtr.Zero, vBuffer, 
技术分享            (uint)vBuffer.Capacity) > 0)
技术分享            return vBuffer.ToString();
技术分享        else return string.Empty;
技术分享    }
技术分享    finally
技术分享    {
技术分享        CloseHandle(vProcess);
技术分享    }
技术分享}
技术分享
技术分享public bool EnumChild(IntPtr hwnd, int lParam)
技术分享{
技术分享    if (!IsWindowVisible(hwnd)) return true; // 不可见
技术分享    StringBuilder vBuffer = new StringBuilder(256);
技术分享    GetClassName(hwnd, vBuffer, vBuffer.Capacity);
技术分享    if (vBuffer.ToString().ToLower() == "richedit20a")
技术分享    {
技术分享        if ((GetWindowLong(hwnd, GWL_STYLE) & ES_READONLY) != ES_READONLY) // 非只读
技术分享        {
技术分享            richHandle = hwnd;
技术分享            return false;
技术分享        }
技术分享    }
技术分享    return true;
技术分享}
技术分享
技术分享public bool EnumFunc(IntPtr hwnd, int lParam)
技术分享{
技术分享    if (!IsWindowVisible(hwnd)) return true; // 不可见
技术分享    StringBuilder vBuffer = new StringBuilder(256);
技术分享    GetClassName(hwnd, vBuffer, vBuffer.Capacity);
技术分享    if (vBuffer.ToString() == "#32770")
技术分享    {
技术分享        uint vProcessId;
技术分享        GetWindowThreadProcessId(hwnd, out vProcessId);
技术分享        if (GetProcessName(vProcessId).ToLower() == "qq.exe")
技术分享        {
技术分享            GetWindowText(hwnd, vBuffer, vBuffer.Capacity);
技术分享            if (vBuffer.ToString().IndexOf("聊天中") >= 0) // 标题中含"聊天中"
技术分享            {
技术分享                EnumChildWindows(hwnd, @EnumChild, lParam);
技术分享                return false;
技术分享            }
技术分享        }
技术分享    }
技术分享    return true;
技术分享}
技术分享
技术分享[StructLayout(LayoutKind.Sequential)]
技术分享public struct GETTEXTLENGTHEX
技术分享{
技术分享    public uint flags;
技术分享    public uint codepage;
技术分享}
技术分享
技术分享[StructLayout(LayoutKind.Sequential)]
技术分享public struct GETTEXTEX
技术分享{
技术分享    public int cb;
技术分享    public int flags;
技术分享    public int codepage;
技术分享    public IntPtr lpDefaultChar;
技术分享    public IntPtr lpUsedDefChar;
技术分享};
技术分享
技术分享public const int GTL_DEFAULT = 0;
技术分享public const int GT_DEFAULT = 0;
技术分享public const int WM_USER = 0x0400;
技术分享public const int EM_GETTEXTEX = WM_USER + 94; 
技术分享public const int EM_GETTEXTLENGTHEX = WM_USER + 95;
技术分享
技术分享public string Process_ReadRichEditText(IntPtr AHandle)
技术分享{
技术分享    if (!IsWindow(AHandle)) return string.Empty;
技术分享    string vReturn = string.Empty;
技术分享    uint vProcessId;
技术分享    GetWindowThreadProcessId(AHandle, out vProcessId);
技术分享    IntPtr vProcess = OpenProcess(PROCESS_VM_OPERATION | PROCESS_VM_READ |
技术分享        PROCESS_VM_WRITE, false, vProcessId);
技术分享    try
技术分享    {
技术分享        uint vNumberOfBytesRead = 0;
技术分享        IntPtr vPointer = VirtualAllocEx(vProcess, IntPtr.Zero, 0x1000, 
技术分享            MEM_RESERVE | MEM_COMMIT, PAGE_READWRITE);
技术分享        GETTEXTLENGTHEX vGetTextLengthEx = new GETTEXTLENGTHEX();
技术分享        vGetTextLengthEx.flags = GTL_DEFAULT;
技术分享        vGetTextLengthEx.codepage = 1200; // Unicode
技术分享        IntPtr vAddress = Marshal.AllocCoTaskMem(Marshal.SizeOf(vGetTextLengthEx));
技术分享        Marshal.StructureToPtr(vGetTextLengthEx, vAddress, false);
技术分享        WriteProcessMemory(vProcess, vPointer, vAddress,
技术分享            Marshal.SizeOf(vGetTextLengthEx), ref vNumberOfBytesRead);
技术分享        Marshal.FreeCoTaskMem(vAddress);
技术分享        int L = SendMessage(AHandle, EM_GETTEXTLENGTHEX, (int)vPointer, 0);
技术分享        VirtualFreeEx(vProcess, vPointer, 0, MEM_RELEASE);
技术分享        if (L <= 0) return vReturn;
技术分享        GETTEXTEX vGetTextEx = new GETTEXTEX();
技术分享        vGetTextEx.cb = L * 2 + 2;
技术分享        vGetTextEx.flags = GT_DEFAULT;
技术分享        vGetTextEx.codepage = 1200; // Unicode
技术分享        vGetTextEx.lpDefaultChar = IntPtr.Zero;
技术分享        vGetTextEx.lpUsedDefChar = IntPtr.Zero;
技术分享        vPointer = VirtualAllocEx(vProcess, IntPtr.Zero,
技术分享            (uint)(Marshal.SizeOf(vGetTextEx) + L * 2 + 2),
技术分享            MEM_RESERVE | MEM_COMMIT, PAGE_READWRITE);
技术分享        vAddress = Marshal.AllocCoTaskMem(Marshal.SizeOf(vGetTextEx));
技术分享        Marshal.StructureToPtr(vGetTextEx, vAddress, false);
技术分享        WriteProcessMemory(vProcess, vPointer, vAddress,
技术分享            Marshal.SizeOf(vGetTextEx), ref vNumberOfBytesRead);
技术分享        Marshal.FreeCoTaskMem(vAddress);
技术分享        SendMessage(AHandle, EM_GETTEXTEX, (int)vPointer,
技术分享            (int)vPointer + Marshal.SizeOf(vGetTextEx));
技术分享        vAddress = Marshal.AllocCoTaskMem(L * 2);
技术分享        ReadProcessMemory(vProcess, 
技术分享            (IntPtr)((int)vPointer + Marshal.SizeOf(vGetTextEx)),
技术分享            vAddress, L * 2, ref vNumberOfBytesRead);
技术分享        vReturn = Marshal.PtrToStringUni(vAddress, L * 2);
技术分享        Marshal.FreeCoTaskMem(vAddress);
技术分享        VirtualFreeEx(vProcess, vPointer, 0, MEM_RELEASE);
技术分享    }
技术分享    finally
技术分享    {
技术分享        CloseHandle(vProcess);
技术分享    }
技术分享    return vReturn;
技术分享}
技术分享
技术分享private void button1_Click(object sender, EventArgs e)
技术分享{
技术分享    richHandle = IntPtr.Zero;
技术分享    EnumWindows(EnumFunc, 0);
技术分享    if (richHandle == IntPtr.Zero) return;
技术分享    Console.WriteLine(Process_ReadRichEditText(richHandle));
技术分享}
技术分享

 
http://blog.csdn.net/zswang/article/details/2009868

获得QQ聊天输入框中的内容