在使用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 %>
答案 0 :(得分:10)
data-disable-with
行为)我们需要在第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;已知问题。
希望在不久的将来你不需要任何解决方法。
答案 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>