也许我们常会有这样的需要,就是当改变主窗口的大小时同时也想他的某些子控件也要随之改变,当然如果数目少或者层次不复杂,那么几个GetWindowRect,MoveWindow定位就能解决问题了,不过如果要处理大量的控件,即一个控件组的改变,或者,属于不同层次(即多重父子依赖关系)的窗口时,就比较麻烦了,这里引用了一个基类,能大大化简以上的工作:
类名CLayoutBase
提供的接口函数有
void AddWinItem(CWnd * inWinItem,CWnd * inParentWnd=NULL);
/inParentWnd为空的时候为默认主窗口的指针
//AddWinItem顺序是有关系的,应该先注册父一级的窗口,然后在注册子一级的窗口
//不然后面的刷新就会先刷新子窗口,后刷新父窗口,不会实现很好的改变效果
void RemoveWinItem(CWnd * inWinItem);
//注销窗口子控件
bool RefreshAllItem();
//由wm_size触发,负责根据改变窗口之后的信息刷新所有注册了的控件
下面介绍储存子窗口信息的结构
struct WinItemInfo
{
CWnd * m_ParentWnd;
CWnd * m_Wnd;
float Xrate;//左上角点X坐标与总界面宽度的比率
float Yrate;//左上角点Y坐标与总界面高度的比率
float Hrate;//元素高度与总界面高度的比率
float Wrate;//元素宽度与总界面宽度的比率
}* m_inWinItem;
储存子窗口队列利用
CObjectList m_WinGroup;
其中CObjectList 为CPtrArray的子类
可以做以下的一些简单的功能扩展
int GetIndex(void * inPointer);
int Add(void * inPointer);
int Add(void * inPointer, int inIndex);
void Remove(void * inPointer);
void * GetPrevious(void * inPointer);
void * GetNext(void * inPointer);
接下来便是接口函数的细节描述了
void CLayoutBase::AddWinItem(CWnd * inWinItem,CWnd * inParentWnd)
{
m_inWinItem = new WinItemInfo;
if(inParentWnd == NULL)
m_inWinItem->m_ParentWnd = ::AfxGetMainWnd( );
else
m_inWinItem->m_ParentWnd = inParentWnd;
RECT m_ItemRect;
m_inWinItem->m_Wnd = inWinItem;
m_inWinItem->m_ParentWnd->GetWindowRect (&m_MainRect);
::GetWindowRect (m_inWinItem->m_Wnd->GetSafeHwnd () ,&m_ItemRect);
m_inWinItem->m_ParentWnd->ScreenToClient(&m_ItemRect);
m_inWinItem->Xrate = (float)(m_ItemRect.left )/(float)(m_MainRect.right-m_MainRect.left);
m_inWinItem->Yrate = (float)(m_ItemRect.top )/(float)(m_MainRect.bottom-m_MainRect.top);
m_inWinItem->Hrate = (float)(m_ItemRect.bottom-m_ItemRect.top)/(float)(m_MainRect.bottom-m_MainRect.top);
m_inWinItem->Wrate = (float)(m_ItemRect.right-m_ItemRect.left)/(float)(m_MainRect.right-m_MainRect.left);
m_WinGroup.Add(m_inWinItem);
}
void CLayoutBase::RemoveWinItem(CWnd * inWinItem)
{
m_WinGroup.Remove(m_inWinItem);
}
bool CLayoutBase::RefreshAllItem()
{
for (int i = 0 ; i <= m_WinGroup.GetSize() - 1 ; i ++)
{
WinItemInfo * receiver = (WinItemInfo *)m_WinGroup.GetAt(i);
if (receiver)
{
receiver->m_ParentWnd->GetWindowRect (&m_MainRect);
//移动背景大小
::MoveWindow(
receiver->m_Wnd ->GetSafeHwnd ()
,(int)((m_MainRect.right -m_MainRect.left)*receiver->Xrate)
,(int)((m_MainRect.bottom -m_MainRect.top)*receiver->Yrate)
,(int)((m_MainRect.right -m_MainRect.left)*receiver->Wrate)
,(int)((m_MainRect.bottom -m_MainRect.top)*receiver->Hrate)
,true);
}
}
//receiver->m_Wnd->SendMessage(WM_ICONERASEBKGND, (WPARAM)receiver->m_Wnd->GetDC ()->GetSafeHdc(), 0);
return true;
}
CLayoutBase的使用方法:
首先把他作为一个窗口类的基类,然后在需要的地方调用AddWinItem()来注册若干个控件,例如:
AddWinItem((GetDlgItem (IDC_BACKGROUND)));
然后在OnSize中调用 RefreshAllItem();即可,而且在需要注销控件的时候就调用RemoveWinItem(CWnd * inWinItem)
另外还需要说明两点:
第一点:AddWinItem()的第一个参数是子控件的指针,第二个参数是他说要参照改变的父控件的指针。这样可以轻松实现多层的父子关系窗口的同步刷新。
第二点:就是函数调用的顺序问题,应该先创建窗口,然后注册窗口,这样他就会自动保存原始的参考位置,最后在进行同步刷新,以便实现正确的拉伸效果。