Zapobieganie podwójnemu wysyłaniu formularzy w jQuery

Mam formularz, który zajmuje trochę czasu, zanim serwer przetworzy. Muszę upewnić się, że użytkownik czeka i nie próbuje ponownie przesłać formularza, klikając przycisk ponownie. Próbowałem użyć następującego kodu jQuery:

<script type="text/javascript">
$(document).ready(function() {
    $("form#my_form").submit(function() {
        $('input').attr('disabled', 'disabled');
        $('a').attr('disabled', 'disabled');
        return true;
    });
});
</script>

Kiedy próbuję to w Firefoksie wszystko zostaje wyłączone, ale formularz nie jest przesyłany z żadnymi danymi postu, które ma zawierać. Nie mogę użyć jQuery do przesłania formularza, ponieważ potrzebuję przycisku do wysłania z formularzem, ponieważ istnieje wiele submit buttons i określić, który został użyty przez którego wartość jest zawarta w poście. Potrzebuję, aby formularz został przesłany tak, jak zwykle i muszę wyłączyć wszystko zaraz po tym.

Dzięki!
Author: smholloway, 2010-05-14

20 answers

Myślę, że Twoim problemem jest ta linia:

$('input').attr('disabled','disabled');
Wyłączasz wszystkie wejścia, w tym te, których dane formularz ma przesłać.

Aby wyłączyć tylko przycisk submit (s), Możesz zrobić to:

$('button[type=submit], input[type=submit]').prop('disabled',true);

Nie sądzę jednak, aby IE przesłało formularz, jeśli nawet te przyciski są wyłączone. Sugerowałbym inne podejście.

Wtyczka jQuery do rozwiązania

Rozwiązaliśmy ten problem za pomocą poniższego kodu. Na sztuczka polega na użyciu data() jQuery, aby oznaczyć formularz jako już złożony lub nie. W ten sposób nie musimy zadzierać z przyciskami submit, co przeraża IE.

// jQuery plugin to prevent double submission of forms
jQuery.fn.preventDoubleSubmission = function() {
  $(this).on('submit',function(e){
    var $form = $(this);

    if ($form.data('submitted') === true) {
      // Previously submitted - don't submit again
      e.preventDefault();
    } else {
      // Mark it so that the next submit can be ignored
      $form.data('submitted', true);
    }
  });

  // Keep chainability
  return this;
};

Użyj go tak:

$('form').preventDoubleSubmission();

Jeśli istnieją formularze AJAX, które powinny być dozwolone do wysyłania wiele razy na ładowanie strony, możesz dać im klasę wskazującą, że, a następnie wykluczyć je z selektora w następujący sposób:

$('form:not(.js-allow-double-submission)').preventDoubleSubmission();
 275
Author: Nathan Long,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2016-04-21 12:46:02

Podejście czasowe jest złe - skąd wiesz, jak długo potrwa akcja na przeglądarce klienta?

Jak to zrobić

$('form').submit(function(){
  $(this).find(':submit').attr('disabled','disabled');
});

Gdy formularz zostanie przesłany, wyłączy wszystkie przyciski Wyślij w środku.

Pamiętaj, że w Firefoksie po wyłączeniu przycisku stan ten zostanie zapamiętany po cofnięciu się do historii. Aby temu zapobiec, musisz na przykład włączyć przyciski podczas ładowania strony.

 38
Author: Ctrl-C,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2013-07-19 07:23:54

Myślę, że odpowiedź Nathana Longa jest właściwą drogą. Dla mnie używam walidacji po stronie klienta, więc dodałem tylko warunek, że formularz będzie ważny.

EDIT: jeśli nie zostanie to dodane, użytkownik nigdy nie będzie mógł przesłać formularza, jeśli Walidacja po stronie klienta napotka błąd.

        // jQuery plugin to prevent double submission of forms
        jQuery.fn.preventDoubleSubmission = function () {
            $(this).on('submit', function (e) {
                var $form = $(this);

                if ($form.data('submitted') === true) {
                    // Previously submitted - don't submit again
                    alert('Form already submitted. Please wait.');
                    e.preventDefault();
                } else {
                    // Mark it so that the next submit can be ignored
                    // ADDED requirement that form be valid
                    if($form.valid()) {
                        $form.data('submitted', true);
                    }
                }
            });

            // Keep chainability
            return this;
        };
 19
Author: PTK,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2015-02-04 22:27:07

event.timeStamp nie działa w Firefoksie. Zwracanie false jest niestandardowe, należy wywołać event.preventDefault(). A skoro już o tym mowa, Zawsze używaj szelek z konstrukcją kontrolną .

Podsumowując wszystkie poprzednie odpowiedzi, oto wtyczka, która wykonuje zadanie i działa między przeglądarkami.

jQuery.fn.preventDoubleSubmission = function() {

    var last_clicked, time_since_clicked;

    jQuery(this).bind('submit', function(event) {

        if(last_clicked) {
            time_since_clicked = jQuery.now() - last_clicked;
        }

        last_clicked = jQuery.now();

        if(time_since_clicked < 2000) {
            // Blocking form submit because it was too soon after the last submit.
            event.preventDefault();
        }

        return true;
    });
};

W odniesieniu do Kern3l, metoda pomiaru czasu działa dla mnie po prostu dlatego, że próbujemy zatrzymać podwójne kliknięcie przycisku Wyślij. Jeśli masz bardzo długi czas odpowiedzi na zgłoszenie, ja zaleca się zastąpienie przycisku Wyślij lub formularza spinnerem.

Całkowite zablokowanie kolejnych przesłań formularza, jak większość powyższych przykładów, ma jeden zły efekt uboczny: jeśli wystąpi awaria sieci i chcą spróbować ponownie przesłać, nie będą w stanie tego zrobić i utracą wprowadzone zmiany. To na pewno zrobi z Ciebie wściekłego użytkownika.

 8
Author: Robin Daugherty,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2017-04-12 07:31:17

Proszę sprawdzić jQuery-safeform plugin.

Przykład użycia:

$('.safeform').safeform({
    timeout: 5000,  // disable next submission for 5 sec
    submit: function() {
        // You can put validation and ajax stuff here...

        // When done no need to wait for timeout, re-enable the form ASAP
        $(this).safeform('complete');
        return false;
    }
});
 7
Author: Maxim Kamenkov,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2015-01-19 23:18:01

...ale formularz nie jest przesyłany z dowolne dane postu, które ma include.

Zgadza się. Wyłączone nazwy/wartości elementów formularza nie będą wysyłane na serwer. Należy ustawić je jako readonly elementy.

Również kotwice nie mogą być wyłączone w ten sposób. Musisz albo usunąć ich href (niezalecane), albo zapobiec ich domyślnemu zachowaniu( lepszy sposób), np.:

<script type="text/javascript">
$(document).ready(function(){
    $("form#my_form").submit(function(){
      $('input').attr('readonly', true);
      $('input[type=submit]').attr("disabled", "disabled");
      $('a').unbind("click").click(function(e) {
          e.preventDefault();
          // or return false;
      });
    });
</script>
 4
Author: karim79,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2010-05-13 22:15:15

Istnieje możliwość poprawy podejścia Nathana Longa . Możesz zastąpić logikę wykrywania już przesłanego formularza tą:

var lastTime = $(this).data("lastSubmitTime");
if (lastTime && typeof lastTime === "object") {
    var now = new Date();
    if ((now - lastTime) > 2000) // 2000ms
        return true;
    else
        return false;
}
$(this).data("lastSubmitTime", new Date());
return true; // or do an ajax call or smth else
 4
Author: Mikhail Fiadosenka,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2017-05-23 11:54:41

Kod Nathana ale dla wtyczki jQuery Validate

Jeśli zdarzy ci się użyć wtyczki jQuery Validate, mają już zaimplementowaną obsługę przesyłania, a w takim przypadku nie ma powodu, aby zaimplementować więcej niż jeden. Kod:

jQuery.validator.setDefaults({
  submitHandler: function(form){
    // Prevent double submit
    if($(form).data('submitted')===true){
      // Previously submitted - don't submit again
      return false;
    } else {
      // Mark form as 'submitted' so that the next submit can be ignored
      $(form).data('submitted', true);
      return true;
    }
  }
});

Możesz go łatwo rozwinąć w } else {-block, aby wyłączyć wejścia i / lub przycisk Wyślij.

Cheers

 3
Author: Alph.Dev,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2015-04-21 09:14:45

W końcu wykorzystałem pomysły z tego postu, aby wymyślić rozwiązanie, które jest dość podobne do wersji AtZako.

 jQuery.fn.preventDoubleSubmission = function() {

    var last_clicked, time_since_clicked;

    $(this).bind('submit', function(event){

    if(last_clicked) 
      time_since_clicked = event.timeStamp - last_clicked;

    last_clicked = event.timeStamp;

    if(time_since_clicked < 2000)
      return false;

    return true;
  });   
};

Używając TAK:

$('#my-form').preventDoubleSubmission();

Odkryłem, że rozwiązania, które nie zawierały pewnego rodzaju limitu czasu, ale tylko wyłączone przesyłanie lub wyłączone Elementy formularza, powodowały problemy, ponieważ po uruchomieniu blokady nie można przesłać ponownie, dopóki nie odświeżysz strony. To powoduje pewne problemy dla mnie podczas robienia rzeczy ajax.

To może być chyba prettied trochę w górę, bo to nie Takie fantazyjne.

 2
Author: jacklin,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2012-06-12 04:07:11

Jeśli używasz AJAX do wysłania formularza, set async: false powinien zapobiegać dodatkowym przesłaniom przed wyczyszczeniem formularza:

$("#form").submit(function(){
    var one = $("#one").val();
    var two = $("#two").val();
    $.ajax({
      type: "POST",
      async: false,  // <------ Will complete submit before allowing further action
      url: "process.php",
      data: "one="+one+"&two="+two+"&add=true",
      success: function(result){
        console.log(result);
        // do something with result
      },
      error: function(){alert('Error!')}
    });
    return false;
   }
});
 2
Author: Al Kari,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2013-06-22 06:01:45

Zmodyfikowano nieco rozwiązanie Nathana dla Bootstrap 3. Spowoduje to ustawienie tekstu wczytywania do przycisku Wyślij. Dodatkowo po upływie 30 sekund formularz zostanie ponownie przesłany.

jQuery.fn.preventDoubleSubmission = function() {
  $('input[type="submit"]').data('loading-text', 'Loading...');

  $(this).on('submit',function(e){
    var $form = $(this);

    $('input[type="submit"]', $form).button('loading');

    if ($form.data('submitted') === true) {
      // Previously submitted - don't submit again
      e.preventDefault();
    } else {
      // Mark it so that the next submit can be ignored
      $form.data('submitted', true);
      $form.setFormTimeout();
    }
  });

  // Keep chainability
  return this;
};

jQuery.fn.setFormTimeout = function() {
  var $form = $(this);
  setTimeout(function() {
    $('input[type="submit"]', $form).button('reset');
    alert('Form failed to submit within 30 seconds');
  }, 30000);
};
 2
Author: AlexZ,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2014-06-07 23:51:19

Użyj dwóch przycisków wyślij.

<input id="sub" name="sub" type="submit" value="OK, Save">
<input id="sub2" name="sub2" type="submit" value="Hidden Submit" style="display:none">

I jQuery:

$("#sub").click(function(){
  $(this).val("Please wait..");
  $(this).attr("disabled","disabled");
  $("#sub2").click();
});
 1
Author: izmir_LEE,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2015-04-18 11:07:30

Miałem podobne problemy i moje rozwiązanie(y) są następujące.

Jeśli nie masz walidacji po stronie klienta, możesz po prostu użyć metody jquery one (), jak to tutaj udokumentowano.

Http://api.jquery.com/one/

To wyłącza funkcję obsługi po jej wywołaniu.

$("#mysavebuttonid").on("click", function () {
  $('form').submit();
});

Jeśli robisz walidację po stronie klienta, tak jak ja, to jest to nieco bardziej skomplikowane. Powyższy przykład nie pozwoli Ci przesłać ponownie po nieudanej walidacji. Spróbuj tego. podejście zamiast

$("#mysavebuttonid").on("click", function (event) {
  $('form').submit();
  if (boolFormPassedClientSideValidation) {
        //form has passed client side validation and is going to be saved
        //now disable this button from future presses
        $(this).off(event);
   }
});
 1
Author: mattbloke,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2015-08-18 15:04:10

Oto Jak to robię:

$(document).ready(function () {
  $('.class_name').click(function () {
    $(this).parent().append('<img src="" />');
    $(this).hide();
  });
});

Napisy: https://github.com/phpawy/jquery-submit-once

 1
Author: phpawy,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2016-07-11 11:41:33

Ten kod wyświetli ładowanie na etykiecie przycisku i ustaw przycisk na

Wyłącz stan, a następnie po przetworzeniu ponownie włącz i zwróć oryginalny tekst przycisku * *

$(function () {

    $(".btn-Loading").each(function (idx, elm) {
        $(elm).click(function () {
            //do processing
            if ($(".input-validation-error").length > 0)
                return;
            $(this).attr("label", $(this).text()).text("loading ....");
            $(this).delay(1000).animate({ disabled: true }, 1000, function () {
                //original event call
                $.when($(elm).delay(1000).one("click")).done(function () {
                    $(this).animate({ disabled: false }, 1000, function () {
                        $(this).text($(this).attr("label"));
                    })
                });
                //processing finalized
            });
        });
    });
    // and fire it after definition
}

);

 0
Author: Mohamed.Abdo,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2014-03-18 15:07:05

Moje rozwiązanie:

// jQuery plugin to prevent double submission of forms
$.fn.preventDoubleSubmission = function () {
    var $form = $(this);

    $form.find('[type="submit"]').click(function () {
        $(this).prop('disabled', true);
        $form.submit();
    });

    // Keep chainability
    return this;
};
 0
Author: ,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2015-08-17 09:54:24

W moim przypadku formularz onsubmit miał jakiś kod weryfikacyjny, więc zwiększam Nathan Long odpowiedź łącznie z punktem kontrolnym onsubmit

$.fn.preventDoubleSubmission = function() {
      $(this).on('submit',function(e){
        var $form = $(this);
        //if the form has something in onsubmit
        var submitCode = $form.attr('onsubmit');
        if(submitCode != undefined && submitCode != ''){
            var submitFunction = new Function (submitCode);
            if(!submitFunction()){
                event.preventDefault();
                return false;
            }                   
        }

        if ($form.data('submitted') === true) {
            /*Previously submitted - don't submit again */
            e.preventDefault();
        } else {
          /*Mark it so that the next submit can be ignored*/
          $form.data('submitted', true);
        }
      });

      /*Keep chainability*/
      return this;
    };
 0
Author: bocapio,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2017-05-23 11:54:41

Zmień przycisk submit:

<input id="submitButtonId" type="submit" value="Delete" />

Z normalnym przyciskiem:

<input id="submitButtonId" type="button" value="Delete" />

Następnie użyj funkcji click:

$("#submitButtonId").click(function () {
        $('#submitButtonId').prop('disabled', true);
        $('#myForm').submit();
    });

I pamiętaj, aby ponownie włączyć przycisk, gdy jest konieczne:

$('#submitButtonId').prop('disabled', false);
 0
Author: Dani,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2017-11-23 15:10:17

Użyj prostego licznika przy wysyłaniu.

    var submitCounter = 0;
    function monitor() {
        submitCounter++;
        if (submitCounter < 2) {
            console.log('Submitted. Attempt: ' + submitCounter);
            return true;
        }
        console.log('Not Submitted. Attempt: ' + submitCounter);
        return false;
    }

I wywołanie funkcji monitor() Podczas przesyłania formularza.

    <form action="/someAction.go" onsubmit="return monitor();" method="POST">
        ....
        <input type="submit" value="Save Data">
    </form>
 0
Author: Mr. Mak,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2018-05-28 08:31:19

Rozwiązałem bardzo podobny problem używając:

$("#my_form").submit(function(){
    $('input[type=submit]').click(function(event){
        event.preventDefault();
    });
});
 -1
Author: Nolan,
Warning: date(): Invalid date.timezone value 'Europe/Kyiv', we selected the timezone 'UTC' for now. in /var/www/agent_stack/data/www/doraprojects.net/template/agent.layouts/content.php on line 54
2011-07-13 19:17:10