hur.cn - 华软网

 热门搜索

Windows程序设计例程求助

  作者:未知    来源:网络    更新时间:2011/1/7
Windows程序设计(第5版)的个例子

/*-------------------------------------------------
   CHECKER4.C -- Mouse Hit-Test Demo Program No. 4
                 (c) Charles Petzold, 1998
  -------------------------------------------------*/

#include <windows.h>

#define DIVISIONS 5

LRESULT CALLBACK WndProc   (HWND, UINT, WPARAM, LPARAM) ;
LRESULT CALLBACK ChildWndProc (HWND, UINT, WPARAM, LPARAM) ;

int   idFocus = 0 ;
TCHAR szChildClass[] = TEXT ("Checker4_Child") ;

int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance,
                    PSTR szCmdLine, int iCmdShow)
{
     static TCHAR szAppName[] = TEXT ("Checker4") ;
     HWND         hwnd ;
     MSG          msg ;
     WNDCLASS     wndclass ;
     
     wndclass.style         = CS_HREDRAW | CS_VREDRAW ;
     wndclass.lpfnWndProc   = WndProc ;
     wndclass.cbClsExtra    = 0 ;
     wndclass.cbWndExtra    = 0 ;
     wndclass.hInstance     = hInstance ;
     wndclass.hIcon         = LoadIcon (NULL, IDI_APPLICATION) ;
     wndclass.hCursor       = LoadCursor (NULL, IDC_ARROW) ;
     wndclass.hbrBackground = (HBRUSH) GetStockObject (WHITE_BRUSH) ;
     wndclass.lpszMenuName  = NULL ;
     wndclass.lpszClassName = szAppName ;
     
     if (!RegisterClass (&wndclass))
     {
          MessageBox (NULL, TEXT ("Program requires Windows NT!"), 
                      szAppName, MB_ICONERROR) ;
          return 0 ;
     }
     
     wndclass.lpfnWndProc   = ChildWndProc ;
     wndclass.cbWndExtra    = sizeof (long) ;
     wndclass.hIcon         = NULL ;
     wndclass.lpszClassName = szChildClass ;
     
     RegisterClass (&wndclass) ;
     
     hwnd = CreateWindow (szAppName, TEXT ("Checker4 Mouse Hit-Test Demo"),
                          WS_OVERLAPPEDWINDOW,
                          CW_USEDEFAULT, CW_USEDEFAULT,
                          CW_USEDEFAULT, CW_USEDEFAULT,
                          NULL, NULL, hInstance, NULL) ;
     
     ShowWindow (hwnd, iCmdShow) ;
     UpdateWindow (hwnd) ;
     
     while (GetMessage (&msg, NULL, 0, 0))
     {
          TranslateMessage (&msg) ;
          DispatchMessage (&msg) ;
     }
     return msg.wParam ;
}

LRESULT CALLBACK WndProc (HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
     static HWND hwndChild[DIVISIONS][DIVISIONS] ;
     int         cxBlock, cyBlock, x, y ;
     
     switch (message)
     {
     case WM_CREATE :
          for (x = 0 ; x < DIVISIONS ; x++)
               for (y = 0 ; y < DIVISIONS ; y++)
                    hwndChild[x][y] = CreateWindow (szChildClass, NULL,
                              WS_CHILDWINDOW | WS_VISIBLE,
                              0, 0, 0, 0,
                              hwnd, (HMENU) (y << 8 | x),
                              (HINSTANCE) GetWindowLong (hwnd, GWL_HINSTANCE),
                              NULL) ;
          return 0 ;
               
     case WM_SIZE :
          cxBlock = LOWORD (lParam) / DIVISIONS ;
          cyBlock = HIWORD (lParam) / DIVISIONS ;
          
          for (x = 0 ; x < DIVISIONS ; x++)
                for (y = 0 ; y < DIVISIONS ; y++)
                    MoveWindow (hwndChild[x][y],
                                x * cxBlock, y * cyBlock,
                                cxBlock, cyBlock, TRUE) ;
          return 0 ;
                       
     case WM_LBUTTONDOWN :
          MessageBeep (0) ;
          return 0 ;

          // On set-focus message, set focus to child window

     case WM_SETFOCUS:
          SetFocus (GetDlgItem (hwnd, idFocus)) ; //这里    
          return 0 ;

          // On key-down message, possibly change the focus window

     case WM_KEYDOWN:
          x = idFocus & 0xFF ;
          y = idFocus >> 8 ;

          switch (wParam)
          {
          case VK_UP:    y-- ;                    break ;
          case VK_DOWN:  y++ ;                    break ;
          case VK_LEFT:  x-- ;                    break ;
          case VK_RIGHT: x++ ;                    break ;
          case VK_HOME:  x = y = 0 ;              break ;
          case VK_END:   x = y = DIVISIONS - 1 ;  break ;
          default:       return 0 ;
          }

          x = (x + DIVISIONS) % DIVISIONS ;
          y = (y + DIVISIONS) % DIVISIONS ;

          idFocus = y << 8 | x ;

          SetFocus (GetDlgItem (hwnd, idFocus)) ;
          return 0 ;

     case WM_DESTROY :
          PostQuitMessage (0) ;
          return 0 ;
     }
     return DefWindowProc (hwnd, message, wParam, lParam) ;
}

LRESULT CALLBACK ChildWndProc (HWND hwnd, UINT message, 
                               WPARAM wParam, LPARAM lParam)
{
     HDC         hdc ;
     PAINTSTRUCT ps ;
     RECT        rect ;
     
     switch (message)
     {
     case WM_CREATE :
          SetWindowLong (hwnd, 0, 0) ;       // on/off flag
          return 0 ;

     case WM_KEYDOWN:
               // Send most key presses to the parent window
          
          if (wParam != VK_RETURN && wParam != VK_SPACE)
          {
               SendMessage (GetParent (hwnd), message, wParam, lParam) ;
               return 0 ;
          }
               // For Return and Space, fall through to toggle the square
          
     case WM_LBUTTONDOWN :
          SetWindowLong (hwnd, 0, 1 ^ GetWindowLong (hwnd, 0)) ;
          SetFocus (hwnd) ;
          InvalidateRect (hwnd, NULL, FALSE) ;
          return 0 ;

               // For focus messages, invalidate the window for repaint
          
     case WM_SETFOCUS:                                    // 这里
          idFocus = GetWindowLong (hwnd, GWL_ID) ;

               // Fall through

     case WM_KILLFOCUS:
          InvalidateRect (hwnd, NULL, TRUE) ;
          return 0 ;
          
     case WM_PAINT :
          hdc = BeginPaint (hwnd, &ps) ;
          
          GetClientRect (hwnd, &rect) ;
          Rectangle (hdc, 0, 0, rect.right, rect.bottom) ;

               // Draw the "x" mark
          
          if (GetWindowLong (hwnd, 0))
          {
               MoveToEx (hdc, 0,          0, NULL) ;
               LineTo   (hdc, rect.right, rect.bottom) ;
               MoveToEx (hdc, 0,          rect.bottom, NULL) ;
               LineTo   (hdc, rect.right, 0) ;
          }

               // Draw the "focus" rectangle
          
          if (hwnd == GetFocus ())
          {
               rect.left   += rect.right / 10 ;
               rect.right  -= rect.left ;
               rect.top    += rect.bottom / 10 ;
               rect.bottom -= rect.top ;

               SelectObject (hdc, GetStockObject (NULL_BRUSH)) ;
               SelectObject (hdc, CreatePen (PS_DASH, 0, 0)) ;
               Rectangle (hdc, rect.left, rect.top, rect.right, rect.bottom) ;
               DeleteObject (SelectObject (hdc, GetStockObject (BLACK_PEN))) ;
          }

          EndPaint (hwnd, &ps) ;
          return 0 ;
     }
     return DefWindowProc (hwnd, message, wParam, lParam) ;
}


书上说:“在CHECKER4中,整体变量idFocus用于保存目前输入焦点窗口的子窗口ID。我在前面说过,当您在子窗口上面单击鼠标时,它们不会自动获得输入焦点。因此,CHECKER4中的父窗口将通过呼叫下面的函数来措置惩罚WM_SETFOCUS消息:
SetFocus (GetDlgItem (hwnd, idFocus)) ;
这样设定一个子窗口为输入焦点。”


2点疑问:
(1)   若父窗口消息措置惩罚程序先措置惩罚WM_setfocus,SetFocus   (GetDlgItem   (hwnd,   idFocus))  中的idFocus 又没有设置好,怎么知道要把输入焦点传给哪一个子窗口? 
    (2)若先是子窗口先措置惩罚WM_setfocus把,     idFocus   =   GetWindowLong   (hwnd,   GWL_ID)   ; 
倒是能够设置idFocus   ,然则这样岂不是已经获得输入焦点了? 且书上说“当您在子窗口上单击时,情况会有些复杂,这时是父窗口而不是子窗口获得输入焦点。”



用鼠标在子窗口上单击时,消息的顺序究竟是如何传送的?




---华软 网友回答---
The WM_SETFOCUS message is sent to a window after it has gained the keyboard focus. 
---华软网友回复---
用VisualStudio自带的spy++工具能够捕捉所有窗口消息,
用用看看就清楚了
---华软网友回复---
引用 2 楼 wonder2008 的回复:
用VisualStudio自带的spy++工具能够捕捉所有窗口消息,
用用看看就清楚了


不会用。。我搜搜看
---华软网友回复---
没人回答了吗
---华软网友回复---
都随便说说啊 结贴呀
---华软网友回复---
(1) 若父窗口消息措置惩罚程序先措置惩罚WM_setfocus,SetFocus (GetDlgItem (hwnd, idFocus)) 中的idFocus 又没有设置好,怎么知道要把输入焦点传给哪一个子窗口?  

首先说下WM_SETFOUCS消息,这个消息是窗口获得了焦点后, 窗口回调函数收到的。请看子窗口回调函数中代码

           case   WM_LBUTTONDOWN :
        
                  SetWindowLong (hwnd, 0, 1 ^ GetWindowLong (hwnd, 0)) ;
                  SetFocus (hwnd) ;
                  InvalidateRect (hwnd, NULL, FALSE) ;
                 return 0 ;
        
                  // For focus messages, invalidate the window for repaint
   
           case   WM_SETFOCUS:
                  idFocus = GetWindowLong (hwnd, GWL_ID) ;

当你按下鼠标键后,消息流程是这样的:子窗口收到WM_LBUTTONDOWN,调用SetFocus将焦点给予该子窗口。子窗口获得焦点后,子窗口回调函数即收到WM_SETFOCUS,措置惩罚WM_SETFOCUS时,你看,idFocus被设置了。不存留你所说"idFocus 又没有设置好"。

你一定会问,即然这里已经设好了子窗口的焦点,那主窗口回调函数中WM_SETFOCUS是做什么用的?

我在前面说过,当您在子窗口上面单击鼠标时,它们不会自动获得输入焦点。因此,CHECKER4中的父窗口将通过呼叫下面的函数来措置惩罚WM_SETFOCUS消息:
SetFocus (GetDlgItem (hwnd, idFocus)) ;
这样设定一个子窗口为输入焦点。”


书上说“当您在子窗口上单击时,情况会有些复杂,这时是父窗口而不是子窗口获得输入焦点。”

我专门尝试了下(用MessageBeep),当我在子窗口上单击时,主窗口的WM_SETFOCUS并未响应。作者是大师,不应该错啊?难道是windows版本问题?作者那会用98,而我现在用XP。

我以为主窗口中的WM_SETFOCUS作用是这样的:
CHECKER4程序窗口最小化、被其它程序窗口遮盖,然后再恢复后是主窗口拥有焦点。所以要措置惩罚WM_SETFOCUS把焦点转给子窗口。


还请高人来继续解答











     
华软声明:本内容来自网络,如有侵犯您版权请来信指出,本站立即删除。