在上一節(jié)中,我簡(jiǎn)單介紹了控件隨父LAYOUT自由移動(dòng)的設(shè)置。在這一節(jié),我將介紹一種常見的情況:嵌入窗口。
在項(xiàng)目中,我們很少會(huì)100%的編寫一個(gè)軟件,特別是界面相關(guān)的,我們會(huì)使用以前已經(jīng)編寫好的窗口,或網(wǎng)上的開源模塊。舉一個(gè)簡(jiǎn)單的例子來說,如果你要編寫一個(gè)視頻播放器,關(guān)于視頻的播放窗口,就用不著用DUI來實(shí)現(xiàn),我們完全可以使用網(wǎng)上的開源庫,嵌入一個(gè)播放的WND即可(當(dāng)然有的庫也支持回調(diào)的方式,用戶可以在自己的窗口中將回調(diào)出來的圖片進(jìn)行自由繪制)。
我們需要在窗口大小改變時(shí),即時(shí)地改變播放窗口的大小。也許你會(huì)說這非常簡(jiǎn)單,直接重載OnSize,然后獲取占位控件(使用占位控件才是最正確的選擇,如果在程序中判斷左邊距、右邊距,就做不到UI、CODE分離了)的大小,然后設(shè)置即可。但是當(dāng)你真正使用的時(shí)候,發(fā)現(xiàn)并沒有那么簡(jiǎn)單。來看代碼:
UIManager.cpp 第750行:
case WM_SIZE: { if( m_pFocus != NULL ) { TEventUI event = { 0 }; event.Type = UIEVENT_WINDOWSIZE; event.pSender = m_pFocus; event.dwTimestamp = ::GetTickCount(); m_pFocus->Event(event); } if( m_pRoot != NULL ) m_pRoot->NeedUpdate(); } return true;
我們看到,窗口大小改變,ROOT只是簡(jiǎn)單的 NeedUpdate,重繪而已,它的大小并沒有設(shè)置為與窗口一樣的大小。
在WinImplBase.cpp 第214 行:
LRESULT WindowImplBase::OnSize(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled){ SIZE szRoundCorner = m_PaintManager.GetRoundCorner();#if defined(WIN32) && !defined(UNDER_CE) if( !::IsIconic(*this) && (szRoundCorner.cx != 0 || szRoundCorner.cy != 0) ) { CDuiRect rcWnd; ::GetWindowRect(*this, &rcWnd); rcWnd.Offset(-rcWnd.left, -rcWnd.top); rcWnd.right++; rcWnd.bottom++; HRGN hRgn = ::CreateRoundRectRgn(rcWnd.left, rcWnd.top, rcWnd.right, rcWnd.bottom, szRoundCorner.cx, szRoundCorner.cy); ::SetWindowRgn(*this, hRgn, TRUE); ::DeleteObject(hRgn); }#endif bHandled = FALSE; return 0;}
也是啥也沒做。
所以在OnSize里面設(shè)置窗口位置,并不會(huì)達(dá)到效果。
那么DUILIB是在哪里設(shè)置ROOT的大小呢?UIManager.cpp 第 615行,即在WM_PAINT中進(jìn)行設(shè)置。
m_pRoot->SetPos(rcClient);
1.SetPos
當(dāng)你看到這里時(shí),我想你已經(jīng)知道第一種方法了。即在 OnSize中,
RECT rc;GetClientRect (m_hWnd, &rc);m_PaintManager.GetRoot()->SetPos (rc);const RECT& rc_pos = targer_ui_->GetPos ();::MoveWindow (move_wnd, rc_pos.left, rc_pos.top, rc_pos.right – rc_pos.left, rc_pos.bottom – rc_pos.top, TRUE);
即我們主動(dòng)設(shè)置大小,ROOT設(shè)置了POS后,會(huì)將它的子控件也設(shè)置POS,詳情請(qǐng)看源碼。所以,我們就能夠得到正確的位置信息了。
但是這并不是最好的方式,原因很簡(jiǎn)單,OnSize會(huì)被頻繁的調(diào)用,特別是在程序初始化的時(shí)候,OnSize被調(diào)用N次,而且在最小化的時(shí)候也會(huì)被調(diào)用。而且當(dāng)你看1.SetPos時(shí),你也猜到了會(huì)有第二種方式了。
2.委托 OnSize
假設(shè)我們的占位控件為 target_ui_,它有一個(gè)委托成員變量:OnSize。直接看代碼吧:
target_ui_->OnSize += MakeDelegate (this, &CYourWnd::OnTargetSizeChanged); bool CYourWnd:: OnTargetSizeChanged (void* param){const RECT& rc_pos = targer_ui_->GetPos ();::MoveWindow (move_wnd, rc_pos.left, rc_pos.top, rc_pos.right – rc_pos.left, rc_pos.bottom – rc_pos.top, TRUE);}
如此簡(jiǎn)單,又如此優(yōu)美的代碼。
注意使用的是 +=。
在這里,我們也看到了作者自己實(shí)現(xiàn)了委托的編寫(我不清楚是不是使用了開源庫),可見作者的C++功底是相當(dāng)深厚的。
看CControlUI的源碼,你會(huì)發(fā)現(xiàn)如下委托對(duì)象:
public: CEventSource OnInit; CEventSource OnDestroy; CEventSource OnSize; CEventSource OnEvent; CEventSource OnNotify;
顧名思義,無需贅述。
這里說一下Event和Notify的區(qū)別。
Event是控件自己收到的消息,比如鼠標(biāo)左鍵按下、彈起、雙擊等,DUILIB先向控件自己發(fā)一個(gè)事件。
Notify通知,是向WND發(fā)送的通知消息,類似MFC中對(duì)話框收到控件的NOTIFY(包括按鈕的單擊),它默認(rèn)情況下是由窗口接收的,在窗口的Notify函數(shù)中進(jìn)行響應(yīng)。
DUILIB的處理流程是,先向CONTROL發(fā)送事件,然后向WND發(fā)送通知。
OnNotify相當(dāng)有用,因?yàn)槟憧梢远ㄖ泼總€(gè)控件的響應(yīng),而不需要在WND的Notify中進(jìn)行一大堆的if..else..了。
OnEvent用處也很大,看情況使用了。
聯(lián)系客服