ASP.NET MVC - How to prevent double click submit with jquery.validate.unobtrusive lib?

后端 未结 9 1115
伪装坚强ぢ
伪装坚强ぢ 2020-12-29 07:47

I need to avoid the double click submitting behavior. I\'m using the client validation with the unobtrusive library. I have the following code for avoiding the double clic:<

相关标签:
9条回答
  • 2020-12-29 07:59

    Based on Ryan P's popular answer I created the following generic solution that also works with my ajax form.

    decorate your custom submit button with the following class:

    <button type="button" class="one-click-submit-button">Submit</button>
    

    Add the following to your javascript file:

    function OneClickSubmitButton() {
        $('.one-click-submit-button').each(function () {
            var $theButton = $(this);
            var $theForm = $theButton.closest('form');
    
            //hide the button and submit the form
            function tieButtonToForm() {
                $theButton.one('click', function () {
                    $theButton.hide();
                    $theForm.submit();
                });
            }
    
            tieButtonToForm();
    
            // This handler will re-wire the event when the form is invalid.
            $theForm.submit(function (event) {
                if (!$(this).valid()) {
                    $theButton.show();
                    event.preventDefault();
                    tieButtonToForm();
                }
            });
        });
    }
    
    OneClickSubmitButton();
    

    since this is an ajax form we want to reload the handlers if we fail server validation.

    function MyForm_OnSuccess() {
        if (true if your form passed validation logic) {
            //do something since your form submitted successfully
        } else { //validation failed on server
            OneClickSubmitButton(); //reinitialize the button logic
        }
    }
    

    Obviously if you don't have ajax forms you can omit the whole OneClickSubmitButton function business and run $('.one-click-submit-button').each(... directly.

    0 讨论(0)
  • 2020-12-29 08:00

    You can also use the JQuery One event.

    I have found that I could get past most guards against double-clicks by double-clicking fast. Using the one event is the only true way to make sure the event is only fired once. I don't think this technique will work "out of the box" with an input type=submit tag. Instead, you can simply use an input type=button or JQueryUI's .button().

    $("#submitButton").one("click", function(event) {
       $('#theForm').submit();
    });
    

    If you need to re-wire the event on a validation error (or other circumstance), I recommend that you create a function for the event handler. The function isn't necessary in this example because all the event handler does is submit the form, but in more complicated scenarios you may want to avoid repeating yourself.

    function submitClick(event) {
       $('#theForm').submit();
    }
    
    $("#submitButton").one('click', function(event) {
       submitClick(event);
    });
    
    // This handler will re-wire the event when the form is invalid.
    $('#theForm').submit(function(event) {
       if (!$(this).valid()) {
          event.preventDefault();
          $('#submitButton').one('click', function(event) { submitClick(event); });
       }
    });
    

    You could obviously add the disabling code here if you wanted to give feedback to the user that the button doesn't work anymore. One great side-effect of using the One event is that you don't actually have to make the button disabled, you can use a style of your own.

    function submitClick(event) {
       $('#submitButton').addClass('disabledButton');
       $('#theForm').submit();
    }
    
    $("#submitButton").one('click', function(event) {
       submitClick(event);
    });
    
    // This handler will re-wire the event when the form is invalid.
    $('#theForm').submit(function(event) {
       if (!$(this).valid()) {
          event.preventDefault();
          $('#submitButton').one('click', function(event) { submitClick(event); });
          $('#submitButton').removeClass('disabledButton');
       }
    });
    

    JQuery One Event: http://api.jquery.com/one/

    0 讨论(0)
  • 2020-12-29 08:04

    I use a different approach to this. Not wiring to the click event of the button, but to the submit event of the form. Works like a charm to prevent multiple simultaneous submits of forms.

    function initFormsToPreventSimultaneousSubmits(selector) {
        if (!selector) {
            selector = 'form'; // No selector supplied, apply to all forms on the page
        }
    
        // Make sure all forms that conform to selector are marked as not submitting
        $(selector).each(function()
        {
            var $form = $(this);
            $form.data('submitting', false);
        });
    
        // Attach to submit event of all forms that conform to selector
        $(selector).off('submit').on('submit', function (e) {
            var $form = $(this);
    
            if (!$form.valid || $form.valid()) { // Make sure to only process when the form is valid or jquery validation is not used
                if ($form.data('submitting')) {
                    // form is already submitting. Classic case of double click on one of the submit buttons of the form. Stop the submit
                    e.preventDefault();
                    return false;
                } else {
                    // All ok, mark the form as submitting and let the form perform the submit
                    $form.data('submitting', true);
                    return true;
                }
            }
        });
    }
    

    On document ready i call initFormsToPreventSimultaneousSubmits() to init all forms on the page.

    Only thing to remember is that when u use a ajax form post is to call the initFormsToPreventSimultaneousSubmits('#formId') on the OnComplete event of the AjaxOptions settings. Because otherwise the form will still be marked as submitting when its done. When a 'normal' form post is used this is not an issue.

    0 讨论(0)
  • 2020-12-29 08:06

    Why not just use:

    function disableButtons() {
        var form = $(this);
        var btns = $("input:submit", form);
    
        if (!form.valid()) {
            // allow user to correct validation errors and re-submit
            btns.removeAttr("disabled");
        } else {
            btns.attr("disabled", "disabled");
        }
    }
    

    to disable your buttons and activate it using:

    $("form").bind("submit", disableButtons);
    
    0 讨论(0)
  • 2020-12-29 08:08

    I have a form that uses MVC3 unobtrusive validation, and a viewmodel with a [RemoteAttribute]. It looks to me like the form's submit event only fires after all validation has passed. I'm currently using this, and it seems to work:

    <input type="submit" value="Submit the Form" 
        data-app-disable-on-submit="true" />
    
    $('form').live('submit', function() {
        $(this).find('input[type="submit"][data-app-disable-on-submit="true"]')
                    .attr('disabled', 'disabled');
    })
    

    ;

    I set breakpoints on both the remote attribute validation action method and the HttpPost action method. Clicking the submit button the first time hits the breakpoint on the validation action method. At this point, the button is still enabled. I can click it multiple times, and after resuming the validation method, the HttpPost is hit only once. When the HttpPost is hit, the submit button is disabled.

    Update

    Right you are Alex. So an updated version of the above would look like this:

    $('form').on('submit', function() {
        $(this).find('input[type="submit"][data-app-disable-on-submit="true"]')
                    .attr('disabled', 'disabled');
    })
    
    0 讨论(0)
  • 2020-12-29 08:11

    I solved it with the following code:

    var tryNumber = 0;
     jQuery('input[type=submit]').click(function (event) {
         var self = $(this);
    
         if (self.closest('form').valid()) {
             if (tryNumber > 0) {
                 tryNumber++;
                 alert('Your form has been already submited. wait please');
                 return false;
             }
             else {
                 tryNumber++;
             }
         };
     });
    

    NOTE: You can also replace the:

    return false;
    

    line, for:

    self.attr('disabled', true);
    

    BUT, if you use the name of your submit buttons on your controller for extra logic, they will be sent as null. (you can use an additional hidden field to charge them before submitting)

    that's it, hope it helps

    Rodrigo

    EDIT: Thanks to these posts: jquery newbie: combine validate with hidding submit button

    0 讨论(0)
提交回复
热议问题