QT源码之Qt创建窗体的过程
创始人
2024-08-01 14:20:26
0

QT源码之Qt创建窗体的过程是本文要介绍的内容。分析Qt的代码也有一段时间了,以前在进行QT源码解析的时候总是使用ue,一个函数名在QTDIR/src目录下反复的查找,然后分析函数之间的调用关系,效率实在是太低了,最近总结出一个更简便的方法,就是利用Qt Creator这个IDE。

带来的好处是:

1、Qt Creator可以很方便的跟踪代码的调用,这样大大提高了分析代码的速度。

2、函数间的调用关系能更加直观的找到。

3、便于对代码的纵向关系的把握。

带来的坏处:

1、只是展现了调用到的函数或者类的关系。

2、缺少对类、某一组类、函数间关系的整体把握。

上面总结一下自己在QT源码解析时候用到的方法,下面开始步入正题。Qt创建窗体的过程,由于我对linux不是很熟悉,下面我所有的分析都是基于windows下的。关于windows下利用API创建窗体。我这里就不多解释了,直接给出代码,然后结合下面的代码来分析一下Qt创建窗体的过程。

详细的解释请参考:

John Chen大牛的博文:WIN32 SDK界面编程

  1. view plaincopy to clipboardprint?  
  2. #include      
  3. LRESULT CALLBACK WndProc (HWND, UINT, WPARAM, LPARAM) ;     
  4. int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance,     
  5.                     PSTR szCmdLine, int iCmdShow)     
  6. {     
  7.     static TCHAR szAppName[] = TEXT ("HelloWin") ;        
  8.          
  9.     HWND         hwnd ;     
  10.     MSG          msg ;     
  11.          
  12.     WNDCLASS     wc ;     
  13.     wc.style         = CS_HREDRAW | CS_VREDRAW ;     
  14.     wc.lpfnWndProc   = WndProc ;     
  15.     wc.cbClsExtra    = 0 ;     
  16.     wc.cbWndExtra    = 0 ;     
  17.     wc.hInstance     = hInstance ;     
  18.     wc.hIcon         = LoadIcon (NULL, IDI_APPLICATION) ;     
  19.     wc.hCursor       = LoadCursor (NULL, IDC_ARROW) ;     
  20.     wc.hbrBackground = (HBRUSH) GetStockObject (WHITE_BRUSH) ;     
  21.     wc.lpszMenuName  = NULL ;     
  22.     wc.lpszClassName = szAppName ;     
  23.     if (!RegisterClass (&wc))     
  24.     {     
  25.         MessageBox (NULL, TEXT ("This program requires Windows NT!"), szAppName, MB_ICONERROR) ;     
  26.         return 0 ;     
  27.              
  28.     }     
  29.     hwnd = CreateWindow (szAppName,                  // window class name     
  30.         TEXT (“hello”), // window caption     
  31.         WS_OVERLAPPEDWINDOW,        // window style     
  32.         CW_USEDEFAULT,              // initial x position     
  33.         CW_USEDEFAULT,              // initial y position     
  34.         CW_USEDEFAULT,              // initial x size     
  35.         CW_USEDEFAULT,              // initial y size     
  36.         NULL,                       // parent window handle     
  37.         NULL,                       // window menu handle     
  38.         hInstance,                  // program instance handle     
  39.         NULL) ;                     // creation parameters     
  40.     ShowWindow (hwnd, iCmdShow) ;     
  41.     UpdateWindow (hwnd) ;     
  42.     while (GetMessage (&msg, NULL, 0, 0))     
  43.     {     
  44.         TranslateMessage (&msg) ;     
  45.         DispatchMessage (&msg) ;     
  46.     }     
  47.     return msg.wParam ;     
  48.          
  49. }     
  50. LRESULT CALLBACK WndProc (HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)     
  51. {     
  52.     HDC         hdc ;     
  53.     PAINTSTRUCT ps ;     
  54.     RECT        rect ;     
  55.     switch (message)     
  56.     {     
  57.     case WM_PAINT:     
  58.         hdc = BeginPaint (hwnd, &ps) ;     
  59.         GetClientRect (hwnd, &rect) ;     
  60.         DrawText (hdc, TEXT ("the WM_PAINTmessage"), -1, &rect,DT_SINGLELINE | DT_CENTER | DT_VCENTER) ;     
  61.         EndPaint (hwnd, &ps) ;     
  62.         return 0 ;     
  63.     case WM_DESTROY:     
  64.         PostQuitMessage (0) ;     
  65.         return 0 ;     
  66.     }     
  67.     return DefWindowProc (hwnd, message, wParam, lParam) ;     
  68. }    
  69. #include  
  70. LRESULT CALLBACK WndProc (HWND, UINT, WPARAM, LPARAM) ;  
  71. int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance,  
  72.      PSTR szCmdLine, int iCmdShow)  
  73. {  
  74.  static TCHAR szAppName[] = TEXT ("HelloWin") ;     
  75.    
  76.  HWND         hwnd ;  
  77.  MSG          msg ;  
  78.    
  79.  WNDCLASS     wc ;  
  80.  wc.style         = CS_HREDRAW | CS_VREDRAW ;  
  81.  wc.lpfnWndProc   = WndProc ;  
  82.  wc.cbClsExtra    = 0 ;  
  83.  wc.cbWndExtra    = 0 ;  
  84.  wc.hInstance     = hInstance ;  
  85.  wc.hIcon         = LoadIcon (NULL, IDI_APPLICATION) ;  
  86.  wc.hCursor       = LoadCursor (NULL, IDC_ARROW) ;  
  87.  wc.hbrBackground = (HBRUSH) GetStockObject (WHITE_BRUSH) ;  
  88.  wc.lpszMenuName  = NULL ;  
  89.  wc.lpszClassName = szAppName ;  
  90.  if (!RegisterClass (&wc))  
  91.  {  
  92.   MessageBox (NULL, TEXT ("This program requires Windows NT!"), szAppName, MB_ICONERROR) ;  
  93.   return 0 ;  
  94.     
  95.  }  
  96.  hwnd = CreateWindow (szAppName,                  // window class name  
  97.   TEXT (“hello”), // window caption  
  98.   WS_OVERLAPPEDWINDOW,        // window style  
  99.   CW_USEDEFAULT,              // initial x position  
  100.   CW_USEDEFAULT,              // initial y position  
  101.   CW_USEDEFAULT,              // initial x size  
  102.   CW_USEDEFAULT,              // initial y size  
  103.   NULL,                       // parent window handle  
  104.   NULL,                       // window menu handle  
  105.   hInstance,                  // program instance handle  
  106.   NULL) ;                     // creation parameters  
  107.  ShowWindow (hwnd, iCmdShow) ;  
  108.  UpdateWindow (hwnd) ;  
  109.  while (GetMessage (&msg, NULL, 0, 0))  
  110.  {  
  111.   TranslateMessage (&msg) ;  
  112.   DispatchMessage (&msg) ;  
  113.  }  
  114.  return msg.wParam ;  
  115.    
  116. }  
  117. LRESULT CALLBACK WndProc (HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)  
  118. {  
  119.  HDC         hdc ;  
  120.  PAINTSTRUCT ps ;  
  121.  RECT        rect ;  
  122.  switch (message)  
  123.  {  
  124.  case WM_PAINT:  
  125.   hdc = BeginPaint (hwnd, &ps) ;  
  126.   GetClientRect (hwnd, &rect) ;  
  127.   DrawText (hdc, TEXT ("the WM_PAINTmessage"), -1, &rect,DT_SINGLELINE | DT_CENTER | DT_VCENTER) ;  
  128.   EndPaint (hwnd, &ps) ;  
  129.   return 0 ;  
  130.  case WM_DESTROY:  
  131.   PostQuitMessage (0) ;  
  132.   return 0 ;  
  133.  }  
  134.  return DefWindowProc (hwnd, message, wParam, lParam) ;  

先写一个最简单的Qt程序 

  1. view plaincopy to clipboardprint?  
  2. #include      
  3. #include      
  4.     
  5. int main(int argc, char *argv[])     
  6. {     
  7.     QApplication a(argc, argv);     
  8.     QPushButton w("hello kitty");     
  9.     w.show();     
  10.     return a.exec();     
  11. }    
  12. #include  
  13. #include  
  14.  
  15. int main(int argc, char *argv[])  
  16. {  
  17.     QApplication a(argc, argv);  
  18.     QPushButton w("hello kitty");  
  19.     w.show();  
  20.     return a.exec();  

来分析一下这个窗体程序是如何创建的。

首先关于main函数和winmain函数,为什么Qt的窗口程序是用main函数而非winmain,在我的另外一篇文章中有解释:

QT源码之QT创建窗口程序、消息循环和WinMain函数  这里不再解释。

Windows窗体创建一定会调用RegisterClass这个函数的,我们在QTDIR/src里面搜索一下,有两个文件有这个函数一个是qapplication_win.cpp另外一个是qeventdispatcher_win.cpp,两个的作用不同,这次我们先研究qapplication_win.cpp中的RegisterClass函数,因为这个是与窗体创建有关的。

下一篇QT源码之Qt是如何处理windows消息 将会介绍qeventdispatcher_win.cpp中的RegisterClass的作用。

我们先将断点设置在qapplication_win.cpp中的 qt_reg_winclass 函数里,然后开始调试,运行到断点,然后我们看一下call stack如下图:

下面红色的框中为Call stack,我们可以看到函数调用的顺序,真正的创建QPushButton是在show()方法中,show()方法又调用了setVisible方法…… ……

QT源码之Qt创建窗体的过程

QtWndProc就是窗体的回调函数,在RegisterClass的时候传给WNDCLASS结构的,QtWndProc同上面的API创建窗体的函数WndProc。

我们看一下QtWndProc的代码,也是一个switch (message) 然后一堆case来处理消息,***也是调用DefWindowProc将不归他处理的消息交还给系统。

小结:QT源码之Qt创建窗体的过程的内容介绍完了,希望本文对你有多帮助!!!

相关内容

热门资讯

如何允许远程连接到MySQL数... [[277004]]【51CTO.com快译】默认情况下,MySQL服务器仅侦听来自localhos...
如何利用交换机和端口设置来管理... 在网络管理中,总是有些人让管理员头疼。下面我们就将介绍一下一个网管员利用交换机以及端口设置等来进行D...
施耐德电气数据中心整体解决方案... 近日,全球能效管理专家施耐德电气正式启动大型体验活动“能效中国行——2012卡车巡展”,作为该活动的...
Windows恶意软件20年“... 在Windows的早期年代,病毒游走于系统之间,偶尔删除文件(但被删除的文件几乎都是可恢复的),并弹...
20个非常棒的扁平设计免费资源 Apple设备的平面图标PSD免费平板UI 平板UI套件24平图标Freen平板UI套件PSD径向平...
德国电信门户网站可实时显示全球... 德国电信周三推出一个门户网站,直观地实时提供其安装在全球各地的传感器网络检测到的网络攻击状况。该网站...
着眼MAC地址,解救无法享受D... 在安装了DHCP服务器的局域网环境中,每一台工作站在上网之前,都要先从DHCP服务器那里享受到地址动...
为啥国人偏爱 Mybatis,... 关于 SQL 和 ORM 的争论,永远都不会终止,我也一直在思考这个问题。昨天又跟群里的小伙伴进行...