用托管C++编写自界说Web组合控件
当前位置:以往代写 > C/C++ 教程 >用托管C++编写自界说Web组合控件
2019-06-13

用托管C++编写自界说Web组合控件

用托管C++编写自界说Web组合控件

副标题#e#

什么是自界说的组合控件

自界说的Web组合控件正如它名字说的那样:在单个控件中集成了一个或多个处事端措施及HTML控件。自界说的组合控件在成果上与用户控件很是雷同,最大的差异之处是,它只存在于它本身的措施会合(或与其他控件共享),能被放在东西条中,并可提供它所包括控件的所见即所得视图方法。

另一方面,自界说Web组合控件比用户控件(user control)越发难建设,因为Visual Studio.NET的设计者们并没有提供可视化建设它们的任何东西,因此,问题是:为什么要用组合控件代替用户控件呢?当分发控件到多个Web措施或系统中时,假如利用自界说Web组合控件,环境要好得多,而用户控件最好用在不重视复用的处所,譬喻,假如只筹备在你本身的网站中利用控件,那么用户控件大概会是更好的选择。根基上来说,你不得不在建设它所花的特别尽力与从中所获得的可复用次数之间,作一衡量;同时,因为自界说组合控件只存在于它本身的措施会合,所以在每台电脑上,只需要一份拷贝,而用户控件则安排于Web措施集内,因此,必需存储在每一个利用它的Web网站上。

建设一个自界说Web组合控件

建设一个自界说Web组合控件的步调,实质上与建设一个自界说的超类Web控件一样,本例中为SearchControl,第一件要做的事,是设计控件的外观,完成之后,看起来大抵如图1所示。

用托管C++编写自定义Web组合控件

图1:设计器中的控件外观

SearchControl,正如上面所看到的,由三个处事端控件构成(实际上有四个,后头将会说到):一个标签控件、一个文本框控件、一个按钮。别的,自界说Web组合控件中较量棘手的部门是它们并没有一个很好的拖放设计东西以支持建设控件,而需要以老方法–手工编写代码来完成。可是,也不完全正确,在此不必手工编写处事端或HTML控件代码,那奈何建设SearchControl的外观呢?

首先,在SearchControl类中写出三个处事端控件的界说:

Label *label;
TextBox *textbox;
Button *button;

接下来,在类的结构函数中建设它们的实例:

SearchControl::SearchControl()
{
 label = new Label();
 textbox = new TextBox();
 button = new Button();
}

最后,在类的CreateChildControls()要领中,把它们添加到一个自界说Web组合控件的子控件荟萃里:

void SearchControl::CreateChildControls()
{
 Controls->Add(label);
 Controls->Add(textbox);
 Controls->Add(button);
}

CreateChildControls()要领是从Control类担任来的一个虚要领,而WebControl也正是从Control类担任而来。

留意,在此并不需要Render()要领,因为构成组合控件的处事端与HTML控件能绘制自身,所以,你完全不消思量此要领,可能在Render()要领中挪用基类:

void SearchControl::Render(HtmlTextWriter *output)
{
 __super::Render(output);
}


#p#副标题#e#

此刻,有了根基的外观了,还可以添加一些成果。首先,需要一些属性,用于更新标签及文本框内的值,以下是属性的界说:

[Bindable(true), Category("Appearance"), DefaultValue("")]
__property void set_Value(String *value);
__property String *get_Value();
[Bindable(true), Category("Appearance"), DefaultValue("")]
__property void set_LabelText(String *value);
__property String *get_LabelText();
[Bindable(true), Category("Appearance"), DefaultValue("")]
__property void set_ButtonText(String *value);
__property String *get_ButtonText();

代码如下:

String *SearchControl::get_LabelText()
{
 this->EnsureChildControls();
 return label->Text;
}
void SearchControl::set_LabelText(String *value)
{
 this->EnsureChildControls();
 label->Text = value;
}
String *SearchControl::get_Value()
{
 this->EnsureChildControls();
 return textbox->Text;
}
void SearchControl::set_Value(String *value)
{
 this->EnsureChildControls();
 textbox->Text = value;
}
String *SearchControl::get_ButtonText()
{
 this->EnsureChildControls();
 return button->Text;
}
void SearchControl::set_ButtonText(String *value)
{
 this->EnsureChildControls();
 button->Text = value;
}

上述代码最棘手的部门就是EnsureChildControls()要领,其担保了子控件在之前已经被建设,假如你不添加这个,设计器将会显示一个空缺的自界说控件。 当你运行上述代码时,将会发明一些设计上的缺陷。首先,文本框老是同样巨细,而且不能分列多于一个SearchControl实例的标签。为举办批改,要添加第四个处事端控件到自界说Web组合控件中,在此,表格(Table)大概是处理惩罚所有控件机关问题最好的要领:

#p#分页标题#e#

void SearchControl::CreateChildControls()
{
 System::Web::UI::WebControls::Table *table = new Table();
 TableRow *row = new TableRow();
 TableCell *cell1 = new TableCell();
 TableCell *cell2 = new TableCell();
 TableCell *cell3 = new TableCell();
 cell1->Controls->Add(label);
 cell2->Controls->Add(textbox);
 cell3->Controls->Add(button);
 row->Cells->Add(cell1);
 row->Cells->Add(cell2);
 row->Cells->Add(cell3);
 table->Rows->Add(row);
 Controls->Add(table);
}

为处理惩罚分列问题,需再再添加两个属性:LabelWidth和LabelAlign。LabelWidth担保了标签控件为一特定的单元长度,而LabelAlign答允标签利用HorizontalAlign列举举办分列:

[Bindable(true), Category("Appearance")]
__property void set_LabelWidth(Unit value);
__property Unit get_LabelWidth();
[Bindable(true), Category("Appearance")]
__property void set_LabelAlign(HorizontalAlign value);
__property HorizontalAlign get_LabelAlign();

#p#副标题#e#

此刻问题又有些棘手了,为指定标签的宽度与分列,可对表格单位格属性举办修改,但在单位格中并不包括标签自身。为简化起见,建设一个名为cellLabel的私有类变量,由其代替CreateChildControls()要领中的cell1,以下是LabelWidth与LabelAlign属性的实现代码:

Unit SearchControl::get_LabelWidth()
{
 this->EnsureChildControls();
 return cellLabel->Width;
}
void SearchControl::set_LabelWidth(Unit value)
{
 this->EnsureChildControls();
 cellLabel->Width = value;
}
HorizontalAlign SearchControl::get_LabelAlign()
{
 this->EnsureChildControls();
 return cellLabel->HorizontalAlign;
}
void SearchControl::set_LabelAlign(HorizontalAlign value)
{
 this->EnsureChildControls();
 cellLabel->HorizontalAlign = value;
}

处理惩罚文本框的缩放时,可使它完全布满包括其的单位格,这样,当表格缩放时,文本框也会随着缩放。于是,当控件缩放时,文本框会随控件变革,可能你可以指定标签为控件的一个百分比,如30%,这样一来,当控件缩放时,标签与文本框城市基于百分比举办调解,以下是修改后的CreateChildControls()要领:

void SearchControl::CreateChildControls()
{
 System::Web::UI::WebControls::Table *table = new Table();
 TableRow *row = new TableRow();
 TableCell *cell2 = new TableCell();
 TableCell *cell3 = new TableCell();
 cellLabel->Controls->Add(label);
 textbox->Width = Unit::Percentage(100);
 cell2->Controls->Add(textbox);
 cell3->Controls->Add(button);
 row->Cells->Add(cellLabel);
 row->Cells->Add(cell2);
 cell3->Width = Unit::Percentage(1);
 row->Cells->Add(cell3);
 table->Rows->Add(row);
 table->Width = Unit::Percentage(100);
 Controls->Add(table);
}

别的,别忘了在结构函数中建设labelCell的实例,要否则可要花点时间找出为什么设计器会给你一个大大的错误提示框了。

处理惩罚事件

此刻,有关控件的外观已经全部完成,是时候对控件插手一点成果了。在此,你想做的第一件事,大概就是让按钮控件能处理惩罚单击事件了,我们在此利用一个事件处理惩罚函数,它就和普通的Windows与Web措施中的一样。以下是在单击事件中添加的事件处理惩罚函数:

button->Click += new EventHandler(this, buttonClicked);

接着,建设一个处理惩罚此事件的要领:

void SearchControl::buttonClicked(Object *sender, EventArgs *e)
{
 OnClick(e);
}

看到了吧,没有多深的科学原理,很简朴。

激发事件

假如想让控件的用户会见到单击事件,那该怎么做呢?我们需要建设一个民众的__event,在此之上,用户可提供他们本身的事件处理惩罚函数:

__event EventHandler* Click;

或者,你还想提供一个受掩护的OnClick()要领,这样,假如有人从SearchControl担任,在传回控件给SearchControl自界说Web组合控件之前或之后,他可为单击事件提供一些特另外成果。别的,假如你很专心,那必然留意到了buttonClick()事件处理惩罚函数实际上是挪用了OnClick()要领,由其把控件从按钮通报到任意事件处理惩罚函数,而事件处理惩罚函数则可被附加于控件的单击事件之上。

void SearchControl::OnClick(EventArgs *e)
{
 if (Click != 0)
  Click(this, e);
}

因为单击事件是默认也是独一的事件,也许,再添加一个DefaultEvent("Click")属性,指定单击为控件的默认事件,会是一个不错的主意:

#p#分页标题#e#

[DefaultProperty("Value"), DefaultEvent("Click"),
ToolboxData("<{0}:SearchControl
runat=server></{0}:SearchControl>")]
public __gc class SearchControl :
public System::Web::UI::WebControls::WebControl
{}

请记着,假如你修改了元数据(属性),那必需删除并从头引用措施集,以便元数据可被从头识别。一般而言,假如要利用此控件,只需在设计器中双击SearchControl控件,以建设一个事件处理惩罚函数。(下面假定利用者的代码为C#代码):

private void SearchControl1_Click(object sender, System.EventArgs e)
{
 WebCustomControl1.Text = SearchControl1.Value;
}

在本文竣事时,作一下简朴的小结,在托管C++中开拓自界说Web控件,涉及了一些较量有意思的工作:首先,它建设了一个由四个处事端控件所构成的自界说Web组合控件,并演示了奈何利用属性来操纵控件;其次,还说明白奈何处理惩罚由控件发生的事件;最后,演示了奈何通报事件给利用自界说组合控件的Web应用措施。

    关键字:

在线提交作业