`

一步步打造防止重复提交按钮

    博客分类:
  • .NET
阅读更多

 原帖及讨论:http://bbs.bccn.net/thread-237267-1-1.html

首先说说防止重复提交按钮是啥东西: 我们在访问有的网站,输入表单完成以后,单击提交按钮进行提交以后,提交按钮就会变为灰色,用户不能再单击第二次,直到重新加载页面或者跳转。这样,可以一定程度上防止用户重复提交导致应用程序上逻辑错误。有朋友说,这个按钮完全可以用js来做,是的。不过当你需要大量这种按钮时,是否为每一个都去编程而不封装一个呢? 另外,为了增加其功能性,我们除了让他有防止重复提交的功能以外,还可以给他弹出提示框,就像单击删除按钮时,用户会受到一个提示再次确认是否真的删除。

好,接下来,第一步:知识储备, 我们知道,要防止重复提交,要在客户端设法使用户单击一次以后按钮变灰,这种客户端行为显然只能借助js代码来完成,服务器端运行的ASP.NET是做不到的。那要让客户端用户单击按钮后立即收到一条再次确认消息,在确认之前不会提交到服务器,也需要js代码。因此,我们基本的丝路就是在页面加载按钮时,一并把所需的js代码发送到客户端去。而该按钮都是适用js脚本,因此不影响服务器端行为。

第二步:从Button继承。 因为它是一个按钮,拥有按钮所需要的全部特征属性,因此我们就从System.Web.UI.WebControls.Button这个类继承。而防止重复提交按钮在按钮变成灰色以后,应该显示什么文本呢?新增一个AfterSubmitText属性来指示;采用一个Bool值“ShowMessageBox”属性来确定是否需要在客户端弹出提示;使用WarningText属性来指示客户端弹出提示的内容。

第三步: 现在,我们就来改写AddAttributesToRender方法,ASP.NET在渲染该控件到输出时,会调用该方法我们所改写的方法,以达到将JS代码发送到客户端的目的。具体代码如下:

       protected override void AddAttributesToRender(HtmlTextWriter writer)
        {
            System.Text.StringBuilder ClientSideEventReference = new System.Text.StringBuilder();

            if (((this.Page != null) && this.CausesValidation) && (this.Page.Validators.Count > 0))
            {
                ClientSideEventReference.Append("if (typeof(Page_ClientValidate) == 'function'){if (Page_ClientValidate() == false){return false;}}");
            }
            //ShowMessageBox?
            if (this.ShowMessageBox)
            {
                ClientSideEventReference.Append("if (!confirm('" + this.WarningText + "')){return false}");
            }
            ClientSideEventReference.AppendFormat("this.value = '{0}';", (string)this.ViewState["afterSubmitText"]);
            ClientSideEventReference.Append("this.disabled = true;");
            ClientSideEventReference.Append(this.Page.ClientScript.GetPostBackEventReference(this, string.Empty));


            writer.AddAttribute(HtmlTextWriterAttribute.Onclick, ClientSideEventReference.ToString(), true);
            base.AddAttributesToRender(writer);
        }

 

 

 

 

 

我们把发送到客户端的js代码看作是一个字符串,为了提高性能,用StringBuilder来对象来构造这个字符串。
首先根据页面对象存在且控件启用了验证(该属性从父继承),且页面对象的验证器内计数大于0来决定输出一段引发验证的js代码,
根据ShowMessageBox属性来决定输出一个弹出提示的代码,并且弹出提示的内容由WarningText属性给出。
书写一个js脚本为按钮赋值为提交后文本,并且将按钮设置为禁用以变灰色。
最后附加一段由ASP.NET提供的用于影射回调事件引用的js脚本(this.Page.ClientScript.GetPostBackEventReference(this, string.Empty)方法将返回一个回送事件引用的js脚本字符串)

将StringBuilder对象内构造的字符串用HtmlTextWriter对象的AddAttribute方法写入按钮的OnClick事件中。

调用父类的AddAttributesToRender方法让父类有机会完成其他的配置等操作。


完整的ClickOnceButton代码如下:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Text;
using System.Web;
using System.Web.UI;
using System.Web.UI.WebControls;

namespace BlogLan.Web.Controls
{
    /// <summary>
    /// 表示一个防止重复提交的按钮。当用户单击按钮以后,该按钮变灰,不能再次单击,直到重新加载页面或者跳转。
    /// </summary>
    [DefaultProperty("Text")]
    [ToolboxData("<{0}:ClickOnceButton runat=server></{0}:ClickOnceButton>")]
    public class ClickOnceButton : System.Web.UI.WebControls.Button
    {
        /// <summary>
        /// 默认的构造函数。
        /// </summary>
        public ClickOnceButton()
        {
            this.ViewState["afterSubmitText"] = "正在提交,请稍候...";
            base.Text = "ClickOnceButton";
            this.ViewState["showMessageBox"] = false;
            this.ViewState["warningText"] = "确定要提交吗?";
        }

        /// <summary>
        /// 获取或设置单击按钮后,按钮上所显示的文本。
        /// </summary>
        [Bindable(true),
        Category("Appearance"),
        DefaultValue("正在提交,请稍候..."),
        Description("指示单击提交后,按钮上所显示的文本。")]
        public string AfterSubmitText
        {
            get
            {
                string afterSubmitText = (string)this.ViewState["afterSubmitText"];
                if (afterSubmitText != null)
                {
                    return afterSubmitText;
                }
                else
                {
                    return string.Empty;
                }
            }
            set
            {
                this.ViewState["afterSubmitText"] = value;
            }
        }

        [Bindable(true),
        Category("Appearance"),
        DefaultValue(false),
        Description("指示是否要显示一个提示框。")]
        public bool ShowMessageBox
        {
            get
            {
                return (bool)this.ViewState["showMessageBox"];
            }
            set
            {
                this.ViewState["showMessageBox"] = value;
            }
        }


        [Bindable(true),
        Category("Appearance"),
        DefaultValue("确定要提交吗?"),
        Description("指示提示框内所包含的内容。")]
        public string WarningText
        {
            get
            {
                return (string)this.ViewState["warningText"];
            }
            set
            {
                this.ViewState["warningText"] = value;
            }
        }

        /// <summary>
        /// AddAttributesToRender
        /// </summary>
        /// <param name="writer">HtmlTextWriter</param>
        protected override void AddAttributesToRender(HtmlTextWriter writer)
        {
            System.Text.StringBuilder ClientSideEventReference = new System.Text.StringBuilder();

            if (((this.Page != null) && this.CausesValidation) && (this.Page.Validators.Count > 0))
            {
                ClientSideEventReference.Append("if (typeof(Page_ClientValidate) == 'function'){if (Page_ClientValidate() == false){return false;}}");
            }
            //ShowMessageBox?
            if (this.ShowMessageBox)
            {
                ClientSideEventReference.Append("if (!confirm('" + this.WarningText + "')){return false}");
            }
            ClientSideEventReference.AppendFormat("this.value = '{0}';", (string)this.ViewState["afterSubmitText"]);
            ClientSideEventReference.Append("this.disabled = true;");
            ClientSideEventReference.Append(this.Page.ClientScript.GetPostBackEventReference(this, string.Empty));


            writer.AddAttribute(HtmlTextWriterAttribute.Onclick, ClientSideEventReference.ToString(), true);
            base.AddAttributesToRender(writer);
        }
    }
}

 

你可以把它编译为dll,放置在工具箱中,随意拖放到网页上即可使用。因为继承了Button控件,它拥有Button的全部特性,并且自动继承了Button的设计时支持。


后记:
当我决定自己开发自定义控件时,这是我第一个想到的,因为明确它就是在一个Button控件基础上,并用JS代码来实现功能。因此,它应该是一个比较简单的东西了。
代码贴出来,与各位分享,尽管现在新技术层出不穷,用AJAX也可以达到效果,不过假如你只是想要防止重复提交,也不用劳烦AJAX这个沉重的框架来完成,有点杀鸡用牛刀了。
愿各位朋友举一反三,深入探讨。

 
 

分享到:
评论

相关推荐

    一步步打造属于自己的ROM

    以下是一步步打造自己ROM的具体步骤和相关知识点: 1. **工具准备**: - **Qoo发布的ROM修改版本**:这是基础ROM,用于构建你自己的版本。 - **nbhtool.exe**:用于处理NBH格式的固件文件,将其分解成可编辑的...

    一步步打造漂亮的新闻列表(PDF)

    【一步步打造漂亮的新闻列表(PDF)】是一系列关于构建高效且美观的新闻列表的文章,主要聚焦于无刷新分页和内容预览功能的实现。该系列由Alexis在Cnblogs博客园撰写,旨在帮助读者理解和应用AJAX技术,提升用户体验...

    一步步学破解.pdf

    一步步学破解.pdf

    一步步打造简单的MVC电商网站BooksStore(4)

    一步步打造一个简单的 MVC 电商网站 – BooksStore(四) 本系列的 GitHub地址:https://github.com/liqingwen2015/Wen.BooksStore 《一步步打造一个简单的 MVC 电商网站 – BooksStore(一)》 《一步步打造一个...

    一步步搭建物联网系统.pdf

    "一步步搭建物联网系统" 本资源是一个关于物联网系统的搭建指南,涵盖了从HTML、Javascript、CSS到GNU/Linux、Arduino、Python、Raspberry Pi等多个技术领域,旨在指导读者一步步搭建自己的物联网系统。 知识点一...

    斯坦福——一步步教你制作计算器软件

    ### 斯坦福大学教学资源:一步步教你制作计算器软件 #### 教程概述 本教程由斯坦福大学提供,旨在教授学生如何开发一个计算器应用程序。该教程为全英文材料,内容简单易懂,并且需要使用Apple电脑及Xcode开发环境...

    Bentley 一步步学习MDL_示例

    在计算机辅助设计(CAD)领域,Bentley MicroStation 是一款广泛使用的软件,它提供了强大的2D和3D建模能力。为了扩展其功能,Bentley 提供了Macro Definition Language (MDL),一种C语言的方言,允许用户自定义工具、...

    seo培训-如何一步步做好SEO-精选文档.ppt

    seo培训-如何一步步做好SEO-精选文档.pptseo培训-如何一步步做好SEO-精选文档.pptseo培训-如何一步步做好SEO-精选文档.pptseo培训-如何一步步做好SEO-精选文档.pptseo培训-如何一步步做好SEO-精选文档.pptseo培训-...

    教你一步步建站

    教你一步步建站 Dreamweaver 是一个功能强大的网站设计和开发工具,本教程将一步步指导您如何使用 Dreamweaver 创建一个网站。 定义站点 1. 在任意一个根目录下创建好一个文件夹(我们这里假设为 E 盘),如取名...

    Exchange 2013 一步步安装图解

    ### Exchange 2013 一步步安装图解 #### 一、准备工作 在正式安装Exchange 2013之前,需要确保服务器环境已经正确设置好。具体步骤包括: 1. **更改计算机名**:为了便于识别和管理,建议将计算机名称更改为有...

    基于opencv一步步实现的车牌识别算法源码+说明.zip

    【资源介绍】 1、该资源包括项目的全部源码,下载可以直接使用! 2、本项目适合作为计算机、数学、电子信息等专业的课程设计、期末大作业和毕设项目,也可以作为...基于opencv一步步实现的车牌识别算法源码+说明.zip

    一步步教你用LoadRunner测试WebService

    一步步教你用LoadRunner测试WebService

    如何一步步学习AJAX

    如何一步步学习AJAX如何一步步学习AJAX如何一步步学习AJAX如何一步步学习AJAX

    《一步步写嵌入式操作系统》书中用到的skyeye

    《一步步写嵌入式操作系统》是一本深入浅出的嵌入式系统开发教材,书中引入了skyeye作为辅助工具,帮助读者理解并实践操作系统的构建过程。skyeye是一款开源的模拟器,它允许开发者在软件层面模拟各种硬件环境,特别...

    图文手把手教你一步步用VC 2010编写通达信缠论插件(2-K线包含处理).pdf

    文档的内容虽然因为OCR扫描的原因存在一些识别错误或漏识别,但整体上通过图文并茂的方式,详细指导了如何一步步完成K线包含处理的编程工作。 编写通达信缠论插件的其他知识还包括对缠论的理解和应用,缠论是一种...

    一步步学习IOS6

    《一步步学习IOS6》文档旨在引导初学者逐步掌握iOS6的开发技巧,通过实例操作,让你深入了解iOS开发的基础知识。本文将重点解析其中的关键概念和步骤。 首先,我们从创建第一个应用程序“HelloWorld”开始。这通常...

Global site tag (gtag.js) - Google Analytics