Reputation: 38652
I have a form that I submit via AJAX.
<%= form_for([@map, @annotation], :remote => true ) do |f| %>
...
<%= f.submit "Save annotation", :class => "btn btn-primary", :id => "annotation-submit-button" %>
<% end %>
I would like to prevent double submits. Since the form only disappears when the request has successfully completed, users can click the submit button as long as the database hasn't finished writing the data. I don't want that, obviously.
I tried adding this to the submit button itself – but it doesn't work. The button gets disabled, but no data is sent.
:onclick => "this.disabled = true"
I've also tried adding a click handler to the submit button. This has the same effect as before. No data is actually sent to the controller, but the button is disabled.
$("#annotation-submit-button").click(function(event) {
$(this).attr("disabled", "disabled");
return false;
});
Also tried the same without returning false
. Nothing happens after the button is disabled.
$("#annotation-submit-button").click(function(event) {
$(this).prop("disabled", "disabled");
});
I begin to think that this is something Rails-specific?
Upvotes: 8
Views: 6469
Reputation: 5213
Try disable_with
in the view like this for Rails 3 and above:
<%= f.submit "Save annotation", :data => {:disable_with => "Saving..."}, ... %>
For Rails 2:
<%= f.submit "Save annotation", :disable_with => "Saving...", ... %>
Upvotes: 20
Reputation: 809
all, After review your ideas and talking in my team, we got an solution:
(function ($) {
/**
* 使用:
* 在jquery的ajax方法中加入参数:beforeSend
* 例如:beforeSend: function(){return $.preventMultipleAjax(event, 5000)}
*
* @param event
* @param delay_duration
* @returns {boolean}
*/
$.preventMultipleAjax = function (event, delay_duration) {
delay_duration = delay_duration || 3000;
var target = $(event.target);
var last_click_time_stamp = target.attr("_ajax_send_time_stamp") || 0;
var time_duration = last_click_time_stamp ? event.timeStamp - last_click_time_stamp : 0;
//console.debug("preventMultipleAjax", last_click_time_stamp, time_duration);
if (time_duration && time_duration < delay_duration) {
return false;
} else {
//console.debug("skip preventMultipleAjax~");
target.attr("_ajax_send_time_stamp", event.timeStamp);
return true;
}
};
/**
* 防止按钮重复点击。
* NOTICE: #1 需要在作用点之前调用此方法 #2 stopImmediatePropagation 会阻止后面的所有事件包括事件冒泡
* @delay_duration 两次点击的间隔时间
*/
$.fn.preventMultipleClick = function (delay_duration) {
delay_duration = delay_duration || 3000;
var last_click_time_stamp = 0;
var time_duration = 0;
$(this).bind('click', function (event) {
time_duration = last_click_time_stamp ? event.timeStamp - last_click_time_stamp : 0;
//console.debug("preventMultipleClick", last_click_time_stamp, time_duration);
if (time_duration && time_duration < delay_duration) {
event.stopImmediatePropagation();
} else {
//console.debug("skip preventMultipleClick~");
last_click_time_stamp = event.timeStamp;
}
});
};
})(jQuery);
If you like it I will added it as an little plugin~ Thanks for your suggestion~
Upvotes: 0
Reputation: 51181
Disabling the button should work just fine.
Just disable the button in the same function where you execute your ajax-call.
$("#annotation-submit-button").click(function(event) {
// do ajax call
$.ajax(...);
// disable the button
$(this).prop("disabled", true);
// prevent the standard action
return false;
});
However, as of jQuery 1.6 you should use prop()
instead of attr()
.
Edit:
In reply to your comment try omitting the return false;
so you don't interrupt the execution of the formsubmit via Rails.
$("#annotation-submit-button").click(function(event) {
// disable the button
$(this).prop("disabled", "disabled");
// do not return false since this stops the event propagation
});
Upvotes: 2
Reputation: 1858
By jQuery: you can do the following to handle the action you want
$.ajax({
.....,
beforeSend: function (){ $('#your-button').attr('disabled', 'disabled'); },
success: function (){
// Here you can enable the button
}
});
Hope this may help you
Upvotes: 1