可擴(kuò)展:
要使得我們的控件具備一定的可擴(kuò)展性,那么必定會(huì)產(chǎn)生控件之外的對(duì)象作為擴(kuò)展,并且這個(gè)對(duì)象對(duì)于控件來說是可插入可移除的。用于擴(kuò)展的對(duì)象和控件之間應(yīng)該具備一定的關(guān)系,例如:1-1,1-n,n-n等。我們將這樣的對(duì)象關(guān)系抽象了出來,稱之為對(duì)象關(guān)系。
對(duì)象關(guān)系:
一個(gè)對(duì)象可能允許單個(gè)對(duì)象對(duì)其進(jìn)行關(guān)聯(lián),也可能允許多個(gè)對(duì)象對(duì)其進(jìn)行關(guān)聯(lián),甚至可能即允許多個(gè)對(duì)象進(jìn)行關(guān)聯(lián),但卻對(duì)某些類型的對(duì)象限制為只能單個(gè)的對(duì)其進(jìn)行關(guān)聯(lián)。我們將這些對(duì)象抽象為:單對(duì)象關(guān)系, 多對(duì)象關(guān)系,獨(dú)占式對(duì)象關(guān)系(這是對(duì)多對(duì)象關(guān)系的一種擴(kuò)展)。

class ObjectRelationship{
protected:
ObjectRelationship(){}
public:
virtual ~ObjectRelationship(){}
private:
ObjectRelationship(const ObjectRelationship&);
ObjectRelationship& operator =(const ObjectRelationship&);
public:
bool CreateRelationship(ObjectRelationship* pObject)
{
if (!DoCreateRelationship_(pObject))
{
return false;
}
RelationshipCreated_(pObject);
return true;
}
bool DestroyRelationship(ObjectRelationship* pObject)
{
if (!DoDestroyRelationship_(pObject))
{
return false;
}
RelationshipDestroyed_(pObject);
return true;
}
protected:
virtual bool DoCreateRelationship_(ObjectRelationship* /*pObject*/) = 0;
virtual bool DoDestroyRelationship_(ObjectRelationship* /*pObject*/) = 0;
protected:
virtual void RelationshipCreated_(ObjectRelationship* /*pObject*/){}
virtual void RelationshipDestroyed_(ObjectRelationship* /*pObject*/){}
};
這是對(duì)象關(guān)系基類,接口只有兩個(gè):建立關(guān)系,銷毀關(guān)系。
單對(duì)象關(guān)系, 多對(duì)象關(guān)系都派生于這個(gè)基類,而獨(dú)占式對(duì)象關(guān)系是實(shí)現(xiàn)的兩個(gè)幫助函數(shù)來輔助多對(duì)象關(guān)系。我們的Widget派生于多對(duì)象關(guān)系,它便具備了和多個(gè)對(duì)象建立關(guān)系的能力(我們將有不同的擴(kuò)展關(guān)聯(lián)到Widget)。為了便于管理和擴(kuò)展,我們將所有和Widget關(guān)聯(lián)的擴(kuò)展放到一個(gè)對(duì)象當(dāng)中進(jìn)行管理,Widget和擴(kuò)展之間的關(guān)系建立和銷毀都委托這個(gè)對(duì)象來進(jìn)行。
class LayoutChildren;
typedef std::set<LayoutChildren*> LayoutChildrenSet;
class RelatedObject{
Widget* pWidget_; // 控件
// 控件所關(guān)聯(lián)的對(duì)象
LayoutChildrenSet layoutChildrens_; // 可以有多個(gè)布局管理管理不同的子控件布局
private:
friend class Widget;
explicit RelatedObject(Widget* const pWidget);
public: // 獲取關(guān)聯(lián)對(duì)象的接口
const LayoutChildrenSet& GetLayoutChildrens() const{return layoutChildrens_;}
private:
void RelationshipCreated_(ObjectRelationship* pObject);
void RelationshipDestroyed_(ObjectRelationship* pObject);
};
void Widget::RelationshipCreated_(ObjectRelationship* pObject)
{
GetRelatedObject()->RelationshipCreated_(pObject);
}
void Widget::RelationshipDestroyed_(ObjectRelationship* pObject)
{
GetRelatedObject()->RelationshipDestroyed_(pObject);
}
布局子控件:
我們?yōu)閃idgt添加了一個(gè)布局子控件的接口,當(dāng)控件自身區(qū)域變化的時(shí)候會(huì)自動(dòng)的調(diào)用這個(gè)接口,當(dāng)然用戶也可以隨時(shí)調(diào)用此接口對(duì)子控件進(jìn)行布局。此接口負(fù)責(zé)將操作傳遞給擴(kuò)展,我們考慮到子控件的布局策略可能會(huì)各有不同,因此我們能夠關(guān)聯(lián)多個(gè)布局子控件擴(kuò)展到Widget,這使得我們能夠以不同的布局策略來區(qū)別對(duì)待不同的子控件。
void Widget::LayoutChildren()
{
auto pRelatedObject = GetRelatedObject();
if (pRelatedObject)
{
const widget::LayoutChildrenSet& layoutChildrens = pRelatedObject->GetLayoutChildrens();
std::for_each(
layoutChildrens.begin(), layoutChildrens.end(),
std::bind(std::mem_fn(&widget::LayoutChildren::Layout), std::placeholders::_1, this));
}
}
我們創(chuàng)建了一個(gè)邊緣式布局自控件擴(kuò)展進(jìn)行測試,測試效果在測試工程中能夠看到。
下載測試工程源碼

作者: Evil.Ghost 發(fā)表于 2011-04-12 21:17 原文鏈接
評(píng)論: 0 查看評(píng)論 發(fā)表評(píng)論
最新新聞:
· 消息稱iPad 3仍不采用Retina屏幕(2011-04-12 20:29)
· nginx 1.0.0發(fā)布(2011-04-12 20:27)
· 2011Mozilla開發(fā)者大會(huì)亮點(diǎn)搶先看(2011-04-12 20:24)
· 亞馬遜成谷歌第一大廣告主 每年2億美元(圖)(2011-04-12 20:16)
· 九城OpenFeint中國首秀:與聯(lián)通推手游(2011-04-12 20:13)
編輯推薦:體驗(yàn)Managed Extensibility Framework精妙的設(shè)計(jì)
網(wǎng)站導(dǎo)航:博客園首頁 我的園子 新聞 閃存 小組 博問 知識(shí)庫