• 给自定义控件(Web Control)添加事件的几种方法。前两种方法可以不实现IPostBackEventHandler



        写自定义控件已经好久了,也有几个用得时间比较长的,但是对于“事件”一直是比较模糊,没有很详细的理解。
        
        最近升级分页控件,由于原来使用的是VB.net(在VB.net里面添加一个事件是比较容易的),现在想改用C#,而原来的方法又写得比较笨拙,想换一个更简洁一点的方法,所以不得不重新认识一下事件。看了几遍《道不远人深入解析asp.net2.0控件开发》居然没有看懂,可能是自己太笨了吧。又找到了自定义分页控件开发 看了几遍,还是比较迷糊,最后把《自定义分页控件开发》 里提供的代码down下来看了一下,终于有点明白了。

        先举一个简单的例子吧,我们建立一个复合控件,在CreateChildControls()里面添加两个控件,一个是Label,一个是LinkButton。我们的目的是要给自定义控件加一个事件,以便可以控制Label的Text属性。

        1、内部事件。
        这个好像是我起的名称,就是只在自定义控件的内部相应的事件,使用控件的页面不用相应相关的事件。具体一点就是要在自定义控件内部,在单击LinkButton的时候修改Label的值。  
    public class EventTest : WebControl, INamingContainer  
        
    {
    protected override void CreateChildControls()
            
    {
                Label lbl 
    = new Label();
                lbl.ID 
    = "lbl";
                
    this.Controls.Add(lbl);
                
    this.Controls.Add(new LiteralControl("<BR>"));
                LinkButton btn 
    = new LinkButton();
                btn.ID 
    = "btn";
                btn.Text 
    = "复合控件的事件测试";
                
    this.Controls.Add(btn);
     }

    }



        编写好这段代码,然后把控件拖拽到页面里面里进行测试,运行后点击LinkButton,我们可以看到页面已经提交了,但是什么变化也没有,恩,我们还没有些事件呢。现在我们就给LinkButton加一个Click的事件。

    protected override void CreateChildControls()
            
    {
                Label lbl 
    = new Label();
                lbl.ID 
    = "lbl"

                
    this.Controls.Add(lbl);
                
    this.Controls.Add(new LiteralControl("<BR>"));

                LinkButton btn 
    = new LinkButton();
                btn.ID 
    = "btn";
                btn.Text 
    = "复合控件的事件测试";
                
    this.Controls.Add(btn);

                
    //给按钮添加内部事件            
                btn.Click += new EventHandler(btn_Click);
    }


     
    void btn_Click(object sender, EventArgs e)
            
    {
                Label lbl 
    = (Label)this.FindControl("lbl");
                lbl.Text 
    += "控件内部的事件,hi";
            }


        再次运行网页,单击,我们看到了想要的效果。

        怎么样简单吧,但是这种方法很不灵活,如果调用控件的网页也想相应事件怎么办呢?

        2、外部事件。
        这回我们要让自定义控件的外部也能相应事件。这个理论上我们要使用委托,也就是delegate,但是系统已经为我们定义好了一个(EventHandler),我们先用现成的。

        这样我们就给自定义控件定义了一个事件,重新编译,我们可以在控件的属性里面看到这个事件。(如果您没有看到,说明没有刷新,可能需要把IDE关闭,再次打开)。
     
    public event EventHandler myClick
            
    {
                add 
    {
                    Events.AddHandler(
    this, value);
                }

                remove 
    {
                    Events.RemoveHandler(
    this, value);
                }

            }


        我们可以添加外部的响应事件了,双击,我们加上几行代码。
    在使用控件的网页里填写的代码 
    protected void EventTest1_myClick(object sender, EventArgs e)
            
    {
                Label lbl 
    = (Label)EventTest1.FindControl("lbl");
                lbl.Text 
    += "外部的事件,Hello" ;

            }



        运行,奇怪还是原来的样子,外部事件并没有相应,不是加了一个事件吗?外面的事件怎么没有被调用呢?这是因为,事件先触发自定义控件内部的事件,然后再由控件内部发出“命令”,调用外部的事件,那么我们怎样才能发出这个命令呢?我们需要要添加这个函数。这个也是最让我费解的地方,可能是对委托不是很了解的原因吧。
     

    protected void TestClick(object sender,EventArgs e)
            
    {
                EventHandler hd 
    = (EventHandler)base.Events[this];
                
    if (hd != null)
                
    {
                    hd(sender, e);
                }

            }


        在修改一下  btn_Click() 就可以了。

    void btn_Click(object sender, EventArgs e)
            
    {
                Label lbl 
    = (Label)this.FindControl("lbl");
                lbl.Text 
    += "控件内部的事件,hi";

                
    //调用外部事件
                TestClick(sender, e);
            }

        再次运行,OK。

        (实现了内部和外部的事件相应,但是这时候我们还没有实现传说中的接口,这是为什么呢?)

        3、“借用”事件,没有成功。

        我们再给自定义控件加上几个<a>标签,让<a>也可以提交表单(就是触发事件),填写如下代码。
     
    protected override void CreateChildControls()
            
    {
    //省略。。。
    this.Controls.Add(new LiteralControl("<BR><a id=\"aa\" href=\"javascript:__doPostBack('EventTest1$btn','')\">aa</a>"));
    }


        运行OK。然后我们修改一下:__doPostBack的第二个参数,
     
    <a id=\"aa\" href=\"javascript:__doPostBack('EventTest1$btn','1')\">aa</a>")


        奇怪,再次运行的时候出现了异常,看来不让这么做呀。

        依赖别人是不行了,必须要实现自定义控件自己的事件了。我们请来 IPostBackEventHandler 帮忙。同时<a>的href修改成
     
    this.Controls.Add(new LiteralControl("<BR><a id=\"a1\" href=\"javascript:__doPostBack('"+ this.ClientID  +"','1')\">[1]</a>"));
                
    this.Controls.Add(new LiteralControl("&nbsp;<a id=\"a2\" href=\"javascript:__doPostBack('" + this.ClientID + "','2')\">[2]</a>"));
                
    this.Controls.Add(new LiteralControl("&nbsp;<a id=\"a3\" href=\"javascript:__doPostBack('" + this.ClientID + "','3')\">[3]</a>"));
                
    this.Controls.Add(new LiteralControl("&nbsp;<a id=\"a4\" href=\"javascript:__doPostBack('" + this.ClientID + "','4')\">[4]</a>"));

    在实现 RaisePostBackEvent 函数 
     
    public void RaisePostBackEvent(string Index)
            
    {
                Label lbl 
    = (Label)this.FindControl("lbl");
                lbl.Text 
    += "自己的事件:" + Index;
            }


        再次运行,OK。同时我们也得到了<a>传递过来的数字(页号)。

        还有一个IPostBackDataHandler接口,由于暂时没不需要,所以没有研究,等用到的时候在研究。
    ----------------------------------------------------------------------

    var theForm = document.forms['form1'];
    if (!theForm) {
        theForm 
    = document.form1;
    }

    function __doPostBack(eventTarget, eventArgument) {
        
    if (!theForm.onsubmit || (theForm.onsubmit() != false)) {
            theForm.__EVENTTARGET.value 
    = eventTarget;
            theForm.__EVENTARGUMENT.value 
    = eventArgument;
            theForm.submit();
        }

    }


        如果您打开IE的“查看源文件”,会看到这个js函数,这是怎么来的呢?这个是button的事件带来的,虽然后面的代码并不需要button了,但是我比较懒,不想自己输出这个js函数,所以还是保留了button的事件。

        
        好像还是说的不太清楚。自定义控件的事件已经整理清楚了,我可以修改分页控件里的代码了,原来写的比较混乱。分页控件的下一个版本(v2.0.0.3)将增加两个事件,这样就可以向吴旗娃的分页控件那样灵活使用了。

    附完整代码:

    namespace JYK.Controls.Page
    {
        [DefaultProperty(
    "Text")]
        [ToolboxData(
    "<{0}:EventTest runat=server></{0}:EventTest>")]
        
    public class EventTest : WebControl , INamingContainer,IPostBackEventHandler  
        
    {
            
    public event EventHandler myClick
            
    {
                add
                
    {
                    Events.AddHandler(
    this, value);
                }

                remove
                
    {
                    Events.RemoveHandler(
    this, value);
                }

            }


            
    protected void TestClick(object sender,EventArgs e)
            
    {
                EventHandler hd 
    = (EventHandler)base.Events[this];
                
    if (hd != null)
                
    {
                    hd(sender, e);
                }

            }


            
    public void RaisePostBackEvent(string Index)
            
    {
                Label lbl 
    = (Label)this.FindControl("lbl");
                lbl.Text 
    += "自己的事件:" + Index;
            }


            
    protected override void CreateChildControls()
            
    {
                Label lbl 
    = new Label();
                lbl.ID 
    = "lbl";

                
    this.Controls.Add(lbl);
                
    this.Controls.Add(new LiteralControl("<BR>"));

                LinkButton btn 
    = new LinkButton();
                btn.ID 
    = "btn";
                btn.Text 
    = "复合控件的事件测试";
                
    this.Controls.Add(btn);

                
    //给按钮添加内部事件

                btn.Click 
    += new EventHandler(btn_Click);

              
    this.Controls.Add(new LiteralControl("<BR><a id=\"aa\" href=\"javascript:__doPostBack('EventTest1$btn','')\">aa</a>"));

                
    this.Controls.Add(new LiteralControl("<BR><a id=\"a1\" href=\"javascript:__doPostBack('"+ this.ClientID  +"','1')\">[1]</a>"));
                
    this.Controls.Add(new LiteralControl("&nbsp;<a id=\"a2\" href=\"javascript:__doPostBack('" + this.ClientID + "','2')\">[2]</a>"));
                
    this.Controls.Add(new LiteralControl("&nbsp;<a id=\"a3\" href=\"javascript:__doPostBack('" + this.ClientID + "','3')\">[3]</a>"));
                
    this.Controls.Add(new LiteralControl("&nbsp;<a id=\"a4\" href=\"javascript:__doPostBack('" + this.ClientID + "','4')\">[4]</a>"));

            }


            
    /// 控件内部的事件,由现有的控件的事件触发
             void btn_Click(object sender, EventArgs e)
            
    {
                Label lbl 
    = (Label)this.FindControl("lbl");
                lbl.Text 
    += "控件内部的事件,hi";

                
    //调用外部事件
                TestClick(sender, e);
            }


            
    protected override void Render(HtmlTextWriter output)
            
    {
                
    if ((base.Site != null&& base.Site.DesignMode)
                
    {
                    output.Write(
    "<div style='TEXT-ALIGN: center;100%'>事件测试</div>");
                }

                
    else
                
    {
                    
    //Page_Click();
                    
    //output.Write("<div id='" + this.ClientID + "Page' style='TEXT-ALIGN: center;90%'>");
                    base.Render(output);
                    
    //output.Write("</div>");
                }

            }

        }

    }




  • 相关阅读:
    iOS遍历程序内某个文件夹下所有文件的属性
    CATransition 转场动画
    Xcode安装的推送证书所在目录
    UIMenuController 实现长按显示自定义菜单功能
    ios调用第三方程序打开文件,以及第三方调用自己的APP打开文件
    购物车界面,不同section,点击增减物品,确定取消选中的逻辑判断
    iOS UINavigationController
    iOS9 URL Schme 白名单
    iOS9 HTTPS
    iOS9 后台定位
  • 原文地址:https://www.cnblogs.com/jyk/p/1235029.html
Copyright © 2020-2023  润新知