Board logo

标题: 如何获取编辑框中的密码 [打印本页]

作者: look_w    时间: 2017-11-18 13:58     标题: 如何获取编辑框中的密码

在众多的共享软件中,有一种小东东可能大家比较感兴趣——《获取编辑框中的密码》。实际上它的实现机理很简单,与获取任何一个窗口中的文字没有本质上的区别。但要获取另一个线程内的编辑框中的密码则稍微有点麻烦,关键是如何获取编辑框的句柄。下面介绍两种方法,供各位有兴趣的朋友参考。
第一种方法:获取鼠标指针法
这种方法的关键是获取当前鼠标指针指向的窗口的句柄。要获取当前鼠标指针的位置使用系统函数::GetCursorPos(&point)即可,参见MSDN中有关cursor的函数。实现步骤如下:
1、在对话框中建立一个定时器(间隔2秒),并在OnTimer函数中作如下操作:
2、在OnTimer(UINT nIDEvent)函数中调用系统函数::GetCursorPos(&point);
3、调用CWnd::WindowFromPoint(point)获取目标窗口的指针w(类型为CWnd*);
5、调用w->GetSafeHwnd( )得到该窗口的句柄hWnd;
6、调用::GetClassName()获取该窗口的类名,判断其类名是否是“Edit”。调用::GetWindowLong(hWnd,GWL_STYLE) 判断其窗口风格是否是ES_PASSWORD,以便确定是不是密码;
7、调用
::SendMessage(
    (HWND) hWnd,              // handle to destination window
      WM_GETTEXT,               // message to send
      (WPARAM)55,          // number of characters to copy
      (LPARAM) sName           // text buffer
    );
获取目标窗口(编辑框)中的文本(密码)。
注意:用::GetWindowText()来获取另一个线程中编辑框中的文字是不行的,必须用SendMessage()。而反之用SendMessage()或::GetWindowText()来获取其他类型窗口的标题却都是可以的。
以下示例的功能是随着鼠标指针的移动,自动获取指针所指向的窗口中的文本(或密码),已调试通过。
void CGetpassDlg::OnTimer(UINT nIDEvent)
{
CPoint point;
::GetCursorPos(&point);
CWnd* w=CWnd::WindowFromPoint(point);

char sName[60];
HWND hWnd=NULL;
if(w!=NULL) hWnd=w->GetSafeHwnd( );

if(hWnd != NULL)
{
  ::GetClassName(hWnd,sName,55);
  CString str(sName);
  
  m_strprompt="这不是密码!(:-<";
  if(str=="Edit" && ::GetWindowLong(hWnd,GWL_STYLE) & ES_PASSWORD)
      m_strprompt="哈哈,密码找到啦!(:-)";
  ::SendMessage(
     (HWND) hWnd,              // handle to destination window
       WM_GETTEXT,               // message to send
       (WPARAM)55,          // number of characters to copy
       (LPARAM) sName           // text buffer
     );
  m_static_pass=sName;  
}
UpdateData(FALSE);

CDialog::OnTimer(nIDEvent);
}
第二种方法:获取输入焦点法
这种方法的关键是获取具有键盘输入焦点的窗口的句柄:
1、在对话框中建立一个定时器(间隔2秒),并在OnTimer函数中作如下操作:
2、如果在Windows 95中,可以先用::GetForegroundWindow()获取用户当前工作的窗口句柄;
3、调用::GetWindowThreadProcessId(hWnd,NULL)获取该窗口所属的线程的标识;
4、调用::GetCurrentThreadId()获取当前线程的标识;
4、调用::AttachThreadInput()将自己的线程与另一个线程关联起来。据我的感觉在这个函数调用中,两个线程标识符在参数表中的先后顺序似乎无所谓;
5、调用:: GetFocus()函数即可获取当前具有键盘输入焦点的窗口句柄;
6、调用::GetClassName()获取该窗口的类名,判断其类名是否是“Edit”。调用::GetWindowLong(hWnd,GWL_STYLE) 判断其窗口风格是否是ES_PASSWORD,以便确定是不是密码;
7、如果该窗口类名是“Edit”,则调用
::SendMessage(
    (HWND) hWnd,              // handle to destination window
      WM_GETTEXT,               // message to send
      (WPARAM)55,          // number of characters to copy
      (LPARAM) sName           // text buffer
    );
8、调用::AttachThreadInput()将自己的线程与另一个线程拆开。
如果在Windows 98 和 Windows NT 4.0 SP3 以上版本中,则用::GetGUIThreadInfo()可以获取系统中当前活动窗口的信息,例如:线程状态、活动窗口句柄、具有键盘输入焦点的窗口句柄、当前捕捉了鼠标输入的窗口句柄、拥有活动菜单的窗口的句柄、当前显示插字符的窗口句柄等。
注意:使用这个函数时,必须包含winable.h文件,而且“#include < winable.h >”指令不要放在stdafx..h中,应直接放在相应的C文件中。
示例代码较长,这里省略,请参阅下面的附录部分。
小结
上述方法实现的程序的特点是短小精悍,只有20多KB,如有朋友需要请来信。我认为第一种比较好,因为它简单。第二种方法用到别的程序里也许用处更大些。您如果有什么好主意,也不妨贡献出来让大家分享。

附录:
下面是第二种方法中OnTimer函数的示例代码(已调试通过):
void  CGetpassDlg::OnTimer(UINT nIDEvent)
{
char sName[60]={0};
//when used for windows 98 or later environment:
#ifdef IN_WIN98_SYSTEM
GUITHREADINFO info;
info.cbSize=sizeof(info);
BOOL bl=::GetGUIThreadInfo(0,&info);
if(bl && info.hwndFocus != NULL)
{
  ::GetClassName(info.hwndFocus,sName,55);
  CString str(sName);
  
  m_strprompt="这不是密码!(:-<";
  if(str=="Edit" && ::GetWindowLong(info.hwndFocus,GWL_STYLE) & ES_PASSWORD)
    m_strprompt="哈哈,密码找到啦!:-)";
  ::SendMessage(
     (HWND) info.hwndFocus,              // handle to destination window
       WM_GETTEXT,               // message to send
       (WPARAM)55,          // number of characters to copy
       (LPARAM) sName           // text buffer
     );
  m_static_pass=sName;  
}
#else
//when used for windows 95 environment:
DWORD curID,wID;
HWND hWnd=::GetForegroundWindow();
if(hWnd!=NULL)
{
  curID=::GetCurrentThreadId();
  wID=::GetWindowThreadProcessId(hWnd,NULL);
  ::AttachThreadInput(
   wID,    // thread to attach
   curID,  // thread to attach to
   TRUE      // attach or detach
   );
  hWnd=::GetFocus();
  if(hWnd!=NULL) ::GetClassName(hWnd,sName,55);
  CString str(sName);
  m_strprompt="这不是密码!:-<";
  if(str=="Edit" && ::GetWindowLong(hWnd,GWL_STYLE) & ES_PASSWORD)
     m_strprompt="哈哈,密码找到啦!:-)";
  ::SendMessage(
    (HWND) hWnd,              // handle to destination window
      WM_GETTEXT,               // message to send
      (WPARAM)55,          // number of characters to copy
      (LPARAM) sName           // text buffer
    );
  m_static_pass=sName;

  ::AttachThreadInput(
   wID,    // thread to attach
   curID,  // thread to attach to
   FALSE      // attach or detach
   );
}
#endif
UpdateData(FALSE);
CDialog::OnTimer(nIDEvent);
}




欢迎光临 电子技术论坛_中国专业的电子工程师学习交流社区-中电网技术论坛 (http://bbs.eccn.com/) Powered by Discuz! 7.0.0