• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            Error

            C++博客 首頁 新隨筆 聯系 聚合 管理
              217 Posts :: 61 Stories :: 32 Comments :: 0 Trackbacks
            首先談談我理解的DirectUI:
                DirectUI是一種策略而不是一種技術,所謂的DirectUI應該是繞開Windows的一套HWND的管理而是自己管理窗口。自己的Event機制,自己的繪圖策略等等等,,,
                如果做的漂亮的話,應該是繪圖API一層,窗口調度和消息派發處理一層,上面再是業務層,,,

                還是先看看UILib的代碼先:

            相關UML:
               
            1.先復習下Windows窗口編程的基礎:
                RegisterWindow -> CreateWindow -> ShowWindow -> UpdateWindow
                最為重要的參數:窗口類。
                窗口類中最重要的成員:窗口回調。
            用上面的流程在追蹤UILib:
            bool CWindowWnd::RegisterWindowClass()
            {
               WNDCLASS wc 
            = { 0 };
               
               wc.lpfnWndProc 
            = CWindowWnd::__WndProc;
               
               ATOM ret 
            = ::RegisterClass(&wc);
               ASSERT(ret
            !=NULL || ::GetLastError()==ERROR_CLASS_ALREADY_EXISTS);
               
            return ret != NULL || ::GetLastError() == ERROR_CLASS_ALREADY_EXISTS;
            }
            復習中提到的最重要的最重要:wc.lpfnWndProc = CWindowWnd::__WndProc;
            代碼對錯誤檢測有冗余,使用ASSERT在測試階段盡可能的暴露bug,但是release一樣能對bug敏感!

            HWND CWindowWnd::Create(HWND hwndParent, LPCTSTR pstrName, DWORD dwStyle, DWORD dwExStyle, int x, int y, int cx, int cy, HMENU hMenu)
            {
               
            if( GetSuperClassName() != NULL && !RegisterSuperclass() ) return NULL;
               
            if( GetSuperClassName() == NULL && !RegisterWindowClass() ) return NULL;
               m_hWnd 
            = ::CreateWindowEx(dwExStyle, GetWindowClassName(), pstrName, dwStyle, x, y, cx, cy, hwndParent, hMenu, CPaintManagerUI::GetResourceInstance(), this);
               ASSERT(m_hWnd
            !=NULL);
               
            return m_hWnd;
            }
            創建窗口之前首先檢查有沒有父窗口,如果有記錄他的部分信息,如果沒有注冊默認窗口類。
            然后調用CreateWindow執行創建窗口的工作。
            然后其它都是在API上做一下簡單的包裝,,,


            接下來看看兩個窗口回調函數:
            LRESULT CALLBACK CWindowWnd::__WndProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
            {
               CWindowWnd
            * pThis = NULL;
               
            if( uMsg == WM_NCCREATE ) {
                  LPCREATESTRUCT lpcs 
            = reinterpret_cast<LPCREATESTRUCT>(lParam);
                  pThis 
            = static_cast<CWindowWnd*>(lpcs->lpCreateParams);
                  pThis
            ->m_hWnd = hWnd;
                  ::SetWindowLongPtr(hWnd, GWLP_USERDATA, reinterpret_cast
            <LPARAM>(pThis));
               } 
               
            else {
                  pThis 
            = reinterpret_cast<CWindowWnd*>(::GetWindowLongPtr(hWnd, GWLP_USERDATA));
                  
            if( uMsg == WM_NCDESTROY && pThis != NULL ) {
                     LRESULT lRes 
            = ::CallWindowProc(pThis->m_OldWndProc, hWnd, uMsg, wParam, lParam);
                     ::SetWindowLongPtr(pThis
            ->m_hWnd, GWLP_USERDATA, 0L);
                     
            if( pThis->m_bSubclassed ) pThis->Unsubclass();
                     pThis
            ->m_hWnd = NULL;
                     pThis
            ->OnFinalMessage(hWnd);
                     
            return lRes;
                  }
               }
               
            if( pThis != NULL ) {
                  
            return pThis->HandleMessage(uMsg, wParam, lParam);
               } 
               
            else {
                  
            return ::DefWindowProc(hWnd, uMsg, wParam, lParam);
               }
            }

            LRESULT CALLBACK CWindowWnd::__ControlProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
            {
               CWindowWnd
            * pThis = NULL;
               
            if( uMsg == WM_NCCREATE ) {
                  LPCREATESTRUCT lpcs 
            = reinterpret_cast<LPCREATESTRUCT>(lParam);
                  pThis 
            = static_cast<CWindowWnd*>(lpcs->lpCreateParams);
                  ::SetProp(hWnd, 
            "WndX", (HANDLE) pThis);
                  pThis
            ->m_hWnd = hWnd;
               } 
               
            else {
                  pThis 
            = reinterpret_cast<CWindowWnd*>(::GetProp(hWnd, "WndX"));
                  
            if( uMsg == WM_NCDESTROY && pThis != NULL ) {
                     LRESULT lRes 
            = ::CallWindowProc(pThis->m_OldWndProc, hWnd, uMsg, wParam, lParam);
                     
            if( pThis->m_bSubclassed ) pThis->Unsubclass();
                     ::SetProp(hWnd, 
            "WndX", NULL);
                     pThis
            ->m_hWnd = NULL;
                     pThis
            ->OnFinalMessage(hWnd);
                     
            return lRes;
                  }
               }
               
            if( pThis != NULL ) {
                  
            return pThis->HandleMessage(uMsg, wParam, lParam);
               } 
               
            else {
                  
            return ::DefWindowProc(hWnd, uMsg, wParam, lParam);
               }
            }
            ::SetProp(hWnd, "WndX", (HANDLE) pThis);
            SetProp為窗口添加一個新的屬性,,,偶覺得沒有必要用這么奇淫的技巧,,,
            當工作在Wnd模式時直接全部是調用自己的窗口回調,當工作在Control模式是調用super回調函數。
            這里的實際上是通過控件的子例化間接的hook了窗口消息。
            在創建窗口的時候完成的子例化窗口類。

            回到分析之前的那個問題:HWND 和 窗口對象是如何關聯起來的?
              LPVOID lpParam        // window-creation data
            使用CreateWindowEx的時候會有一個機會提供創建參數,回調中的WM_NCCREATE會響應它,,,然后在這里把窗口句柄摳出來,再使用SetWindowLog關聯到窗口句柄上,,,






            posted on 2011-06-15 16:57 Enic 閱讀(4991) 評論(0)  編輯 收藏 引用
            久久se精品一区二区影院| 77777亚洲午夜久久多喷| 久久久久久综合一区中文字幕| 青青草原1769久久免费播放| 亚洲精品tv久久久久| 欧美熟妇另类久久久久久不卡| 观看 国产综合久久久久鬼色 欧美 亚洲 一区二区 | 国产精品久久久久aaaa| 亚洲午夜福利精品久久| 国产精品久久久久aaaa| 久久精品日日躁夜夜躁欧美| 国产ww久久久久久久久久| 久久国产乱子伦免费精品| 国产精品99久久久久久宅男小说| 7国产欧美日韩综合天堂中文久久久久| 亚洲精品97久久中文字幕无码| 亚洲嫩草影院久久精品| 97久久综合精品久久久综合| 精品熟女少妇AV免费久久| 亚洲AV伊人久久青青草原| 国産精品久久久久久久| 中文精品久久久久国产网址| 国产精品久久永久免费| 国产精品免费福利久久| 国产69精品久久久久9999APGF| 亚洲国产成人精品女人久久久| 国产精品久久午夜夜伦鲁鲁| 欧美精品久久久久久久自慰| 亚洲午夜久久久影院伊人| 久久精品一区二区三区AV| 久久这里的只有是精品23| 日韩精品久久久久久久电影| 色婷婷狠狠久久综合五月| 久久久久亚洲AV成人网人人软件| 国产精品免费久久久久久久久| 亚洲国产精品一区二区久久| 亚洲欧美精品伊人久久| 久久午夜福利电影| 久久久久亚洲AV无码专区首JN| 99精品久久久久久久婷婷| 日产精品久久久一区二区|