如何在远程表单上禁用提交按钮,直到加载下一页

时间:2017-10-19 13:49:59

标签: javascript ruby-on-rails turbolinks ujs

在使用Turbolinks的Rails 5.1应用程序中,我在提交按钮中添加了data-disable-with属性,因此单击时,该按钮将被禁用,以防止意外多次提交数据。这在很多情况下都很有效。

问题是,在使用内置的UJS助手(data-remote=true)通过AJAX提交的表单上,单击提交按钮时,它不会保持禁用状态。它最初被禁用,但在下一页加载之前会快速重新启用。这违背了data-disable-with行为的要点,因为它可以重新提交意外形式。

有没有办法在新页面加载之前保持表单按钮被禁用?

以下是表格:

<%= simple_form_for(
  resource,
  as: resource_name,
  url: session_path(resource_name),
  html: { class: "large", autocomplete: "on" },
  remote: true
) do |f| %>
  <%= f.input(
    :email,
    placeholder: "Email address",
    label: false,
    autofocus: true
  ) %>
  <%= f.input(:password, placeholder: "Password", label: false) %>
  <%= f.button(
    :submit,
    "Sign in",
    class: "ui fluid large teal submit button",
    "data-disable-with": "Signing in..."
  ) %>
<% end %>

这里发生了什么。 Example of submit button disabling properly on click, but then re-enabling before new page is loaded

3 个答案:

答案 0 :(得分:10)

发生了什么事?

  1. 表格已提交
  2. rails-ujs禁用按钮(data-disable-with行为)
  3. 表单请求成功
  4. rails-ujs重新启用按钮
  5. turbolinks-rails向重定向位置发出请求(可能是一个缓慢的位置,使按钮处于启用状态)
  6. 解决方案

    我们需要在第4步之后重新停用该按钮。为此,我们会听取ajax:success事件,并使用setTimeout禁用它。这确保了在 Rails完成它之后它将被禁用。 (您可以使用requestAnimationFrame代替setTimeout,但它不受广泛支持。)

    为了防止按钮被缓存在禁用状态,我们会在缓存之前重新启用它。 (注意使用one而不是on来防止before-cache处理程序执行多次。)

    我注意到你使用的是jQuery和jquery-ujs,所以我将在下面的代码中使用这些库中的函数。将其包含在您的主JavaScript文件中。

    <强>的jquery-UJS

    ;(function () {
      var $doc = $(document)
    
      $doc.on('submit', 'form[data-remote=true]', function () {
        var $form = $(this)
        var $button = $form.find('[data-disable-with]')
        if (!$button.length) return
    
        $form.on('ajax:complete', function () {
          // Use setTimeout to prevent race-condition when Rails re-enables the button
          setTimeout(function () {
            $.rails.disableFormElement($button)
          }, 0)
        })
    
        // Prevent button from being cached in disabled state
        $doc.one('turbolinks:before-cache', function () {
          $.rails.enableFormElement($button)
        })
      })
    })()
    

    rails-ujs / jQuery

    ;(function () {
      var $doc = $(document)
    
      $doc.on('ajax:send', 'form[data-remote=true]', function () {
        var $form = $(this)
        var $button = $form.find('[data-disable-with]')
        if (!$button.length) return
    
        $form.on('ajax:complete', function () {
          // Use setTimeout to prevent race-condition when Rails re-enables the button
          setTimeout(function () {
            $button.each(function () { Rails.disableElement(this) })
          }, 0)
        })
    
        // Prevent button from being cached in disabled state
        $doc.one('turbolinks:before-cache', function () {
          $button.each(function () { Rails.enableElement(this) })
        })
      })
    })()
    

    rails-ujs / vanilla JS

    Rails.delegate(document, 'form[data-remote=true]', 'ajax:send', function (event) {
      var form = event.target
      var buttons = form.querySelectorAll('[data-disable-with]')
      if (!buttons.length) return
    
      function disableButtons () {
        buttons.forEach(function (button) { Rails.disableElement(button) })
      }
    
      function enableButtons () {
        buttons.forEach(function (button) { Rails.enableElement(button) })
      }
    
      function beforeCache () {
        enableButtons()
        document.removeEventListener('turbolinks:before-cache', beforeCache)
      }
    
      form.addEventListener('ajax:complete', function () {
        // Use setTimeout to prevent race-condition when Rails re-enables the button
        setTimeout(disableButtons, 0)
      })
    
      // Prevent button from being cached in disabled state
      document.addEventListener('turbolinks:before-cache', beforeCache)
    })
    

    请注意,这会禁用按钮,直到所有data-remote表单上的下一页加载data-disable-with按钮为止。您可能希望更改jQuery选择器以仅将此行为添加到选定的表单。

    希望有所帮助!

答案 1 :(得分:2)

仅供参考:Rails&#34; rails-ujs prematurely enables disabled elements for XHR requests with redirects&#34;已知问题。

还有两个PR要修复它:#1#2

希望在不久的将来你不需要任何解决方法。

答案 2 :(得分:0)

如何提交表单数据和禁用按钮。

<form method="post" enctype="multipart/form-data" action="action.php">
    <div class="panel panel-primary">
        <div class="panel-heading">
            <h3 class="panel-title"> submit and disabled button </h3>
        </div>
        <div class="panel-body">
            {% csrf_token %}
            <div class="form-group">
                <label>Montant</label>
                <input type="number" name="montant" id="montant" required/>
            </div>
            <div class="form-group">
                <center><span id="msgError"></span></center>
            </div>
        </div>
    </div>
<div class="form-group">
    <button type="submit" name="save" id="save">Enregistrer</button>
</div>

</form>`

<script>

    $(document).ready(function(){
        var count = 0;
        $('#save').click(function(){
           var montant = $('#montant').val();
           if (montant!=''){
               count++;
               var montant = $('#montant').val();
               // first click data are sending
               if (count == 1){
                  return true;
               }else{
               // disable button 
                  $('#msgError').text('Merci de patienter...');
                  $('#msgError').css('color', 'blue');
                  $('#save').attr('disabled', 'disabled');
               }
           }
        });
   });
</script>