How to access Rails model attributes inside JavScript? - javascript

<%= form_for(#mymodel, remote: true, html: { id: 'match_form' }) do |f| %>
<!-- I need to check if #mymodel.match_id matches the value generated by a controller function -->
<%= f.submit 'Save', class: 'btn btn-primary', id: 'match_submit', style: "width:38px;padding:0px" %>
<%= button_tag 'Cancel', class: 'btn btn-secondary', id: 'match_cancel', style: "width:52px;padding:0px" %>
<% end%>
<script type='text/javascript'>
$(function() {
$(document).on("click", "#match_submit", function(event){
$.ajax('my_controller_method', {
type: 'GET',
dataType: 'script',
data: {
mid: $("#").val(), // how do I pass #mymodel.match_id here?
},
error: function(jqXHR, textStatus, errorThrown) {
return console.log("AJAX Error: " + textStatus);
}
});
});
</script>
I have a Rails form that represents a model as shown above.
How can I access attributes of the model inside the JavaScript code block shown above?

You can use erb inside of javascript too:
mid: <%= #mymodel.match_id %>,
Or you can serialize your object with gon gem (https://github.com/gazay/gon)

Related

How to prevent sending more than one request while waiting ajax to execute?

I do have an anchor which, when pressed is sending an ajax request to swap element and while waiting it to execute i can click it one more time and send another request. How to prevent this situation?
<% if current_or_guest_user.follow?(id_to_follow) %>
<%= link_to raw('<span class="glyphicon glyphicon-eye-close"></span> Unfollow'), unfollow_path(follow.id), method: "delete", remote: true,
class: 'btn btn-warning follow-link', id: 'unfollow'%>
<% else %>
<%= link_to raw('<span class="glyphicon glyphicon-eye-open"></span> Follow'), follow_path(following_id: id_to_follow), method: "post", remote: true,
class: 'btn btn-success follow-link', id: 'follow' %>
<% end %>
$(document).on('ajax:success',"#unfollow", function(xhr, data){
$(this).replaceWith(data['follow']);
});
$(document).on('ajax:success', "#follow", function(xhr, data){
$(this).replaceWith(data['follow']);
});

Rails/Ajax/ jQuery: Pass multiple params from jquery to Rails controller

I need to pass two params (selected option & selected number of persons) from radio buttons through jquery to Rails controller via ajax and would appreciate any help at all on how to do so. Both params are needed to calculate pricing (please see #price code in controller below) and right now I'm only able to pass one param at a time for the calculation.
My form html code is shown below:
<%= form_for #cartedactivity, url: carted_activities_path, method: :post, remote: true do |f| %>
1. Choose option
<div class = "options">
<center>
<% #options.each do |option| %>
<div class = "radio" id = "option_radio">
<%= f.radio_button :supplier_activity_option_id, option.id, :onclick => "save_option();" %>
<%= f.label :supplier_activity_option_id, option.option_name, :value => option.id %>
</div>
<% end %>
</center>
</div>
<h5>2. Number of Persons</h5>
<div class = "numbers">
<center>
<h5>
<%#numbers.each do |number| %>
<div class = "radio" id =
"quantity_radio">
<%= f.radio_button :activity_quantity, number, :onclick => "save_quantity();" %>
<%= f.label :activity_quantity, number, :value => number %>
</div>
<% end %>
</h5>
</center>
</div>
<h5>3. Start date
<%= f.date_field :activity_date, min: (Date.today+2.days), max: (Date.today+4.months)%>
</h5>
<hr>
<div id = "check-price-<%=#option%>"></div>
<div class = "booking-btn">
<%= f.submit "Add to Cart", class: "btn btn-book" %>
</div>
<% end %>
</div>
</div>
<br><br>
<script>
function save_option()
{var optionValue = $('input[name = "carted_activity[supplier_activity_option_id]"]:checked').val();
if (optionValue) {
$.ajax({
type: "POST",
url: "/supplier_activities/price_check",
data: { optionValue },
success: function(post){ console.log('success') },
error: function(post){ console.log(this) }
});
}
};
function save_quantity()
{var quantityValue = $('input[name = "carted_activity[activity_quantity]"]:checked').val();
if (quantityValue) {
$.ajax({
type: "POST",
url: "/supplier_activities/price_check",
data: { quantityValue },
success: function(post){ console.log('success') },
error: function(post){ console.log(this) }
});
}
};
The partial html is simply <% #price %>
The controller code is :
def price_check
#option = params[:optionValue]
#number = params[:quantityValue]
if !#option.blank? && !#number.blank?
#price = Price.find_by("supplier_activity_option_id = ? AND min_qty_for_price <= ? AND max_qty_for_price >= ?", params[:optionValue], params[:quantityValue], params[:quantityValue]).price_usd
else
#price = 100
end
respond_to do |format|
format.js {render layout: false}
end
end
Found the solution - have passed through the value of both checked radio buttons on each click of any radio button. This means that whenever a radio button is clicked, the values of both checked buttons are passed in at the same time.
<script>
function save_option()
{var optionValue = $('input[name = "carted_activity[supplier_activity_option_id]"]:checked').val();
var quantityValue = $('input[name = "carted_activity[activity_quantity]"]:checked').val();
if (optionValue) {
$.ajax({
type: "POST",
url: "/supplier_activities/price_check",
data: { optionValue: optionValue, quantityValue: quantityValue },
success: function(post){ console.log('success') },
error: function(post){ console.log(this) }
});
}
};
function save_quantity()
{var quantityValue = $('input[name = "carted_activity[activity_quantity]"]:checked').val();
var optionValue = $('input[name = "carted_activity[supplier_activity_option_id]"]:checked').val();
if (quantityValue) {
$.ajax({
type: "POST",
url: "/supplier_activities/price_check",
data: { quantityValue: quantityValue, optionValue: optionValue },
success: function(post){ console.log('success') },
error: function(post){ console.log(this) }
});
}
};
</script>

Rails: remote: true from select_tag

I'm calling an AJAX function from a select_tag like so:
<%= select_tag 'quantity', options_from_collection_for_select(order.options), :quantity, :quantity, order.quantity), onchange: "update_price(#{order.id}, this.value);" %>
And here's the function:
<script type='text/javascript'>
function update_price(order_id, quantity) {
$.ajax({
url: "/cart/" + <%= #cart_transaction.id %> + "/update_quantity",
type: "POST",
data: {
"order_id" : order_id,
"quantity" : quantity },
dataType: "html"
});
}
</script>
My .js.erb isn't called ever, and I suspect it's because I haven't specified remote: true anywhere, but since I don't have a form per se I don't know how to do that. Any help?
Relevant controller code here:
class CartTransactionsController < ApplicationController
load_and_authorize_resource
respond_to :html, :js
before_filter :set_cart_transaction
def update_quantity
#order = #cart_transaction.orders.find(params[:order_id])
#price = current_user.brand.prices
.where(template_id: #order.document.template.id)
.where(quantity: params[:quantity]).first
#order.update_attributes(
price_cents: #price.amount_cents, quantity: params[:quantity]
)
#cart_transaction.save!
respond_to { |format| format.js }
end
private
def set_cart_transaction
#cart_transaction = current_user.cart
end
def cart_transactions_params
params.require(:cart_transaction).permit(
:name, :email, :delivery_address, :comments
)
end
end
Update
Here's the .js.erb that isn't called for some reason:
console.log("update_quantity.js.erb file");
$('#price_cell').html("<%= j render(partial: 'price', locals: { order: #order }) %>");
$('#subtotals').html("<%= j render(partial: 'subtotals', locals: { cart_transaction: #cart_transaction }) %>");
Try this:
function update_price(order_id, quantity) {
$.ajax({
beforeSend: function(xhr) {
xhr.setRequestHeader('X-CSRF-Token', $('meta[name="csrf-token"]').attr('content'));
},
url: "/cart/" + <%= #cart_transaction.id %> + "/update_quantity",
type: "POST",
data: {
"order_id" : order_id,
"quantity" : quantity }
});
}
Use dataType: "script", it will work and will render js.erb

Dynamic Select in RubyOnRails error in append select

I'm trying to do a dynamic select, when user select school appear only the groups that belong to that school, but when I tried to change the select I got this error:
Missing partial admin/groups/_group with {:locale=>[:en], :formats=>[:js, :html], :variants=>[], :handlers=>[:erb, :builder, :raw, :ruby, :coffee]}.
CoffeeScript
$(document).on "page:change", ->
$(document).on 'change', '#schools_select', (evt) ->
$.ajax 'update_groups',
type: 'GET'
dataType: 'script'
data: {
school_id: $("#schools_select option:selected").val()
}
error: (jqXHR, textStatus, errorThrown) ->
console.log("AJAX Error: #{errorThrown}")
success: (data, textStatus, jqXHR) ->
console.log("Dynamic school select OK!")
HTML
The school belongs to the model of the form, but the group isn't an attribute of the form.
<div class="form-group">
<%= f.label :school %><br>
<%= f.select :school_id, options_for_select(#schools.active.collect { |school|
[school.name.titleize, school.id] }, 1), {}, { id: 'schools_select' } %>
</div>
<div class="form-group">
<%= label_tag :group %><br>
<%= select_tag :group_id, options_for_select(#groups.active.collect { |group|
[group.name.titleize, group.id] }, 1), { id: 'groups_select' } %>
</div>
Javascript
$("#groups_select").empty().append("<%= escape_javascript(render(:partial => #groups)) %>")
Ruby
def update_groups
#groups = Group.where(school_id: params[:school_id])
respond_to do |format|
format.js
end
end
Routes
get 'students/update_groups', as: 'update_groups'
Make your method like below. Also, make sure yr ajax request triggers.
def update_groups
#groups = Group.where(school_id: params[:school_id])
render json: #groups, root: false
end
My error was that I didn't tell what to render and then I was trying to update the select in other file, when it wasn't necessary.
This is the updated code
CofeeScript
$(document).on "page:change", ->
$(document).on 'change', '#schools_select', (evt) ->
$.ajax 'update_groups',
type: 'GET'
dataType: 'script'
data: {
school_id: $("#schools_select option:selected").val()
}
error: (jqXHR, textStatus, errorThrown) ->
console.log("AJAX Error: #{errorThrown}")
success: (data, textStatus, jqXHR) ->
groups = JSON.parse(data)
$.each groups, (index) ->
$('#groups_select').empty().append $('<option></option>').attr('value', groups[index].id).text(groups[index].name)
return
Ruby
def update_groups
#groups = Group.where(school_id: params[:school_id])
respond_to do |format|
format.json { render json: #groups }
end
end
I delete the javascript file.

Updating vote counter rails ajax request

I have a rails app, in which my Posts model has Comments and the comments are votable. I'm using acts_as_votable.
I currently have the voting on the comments working. Now I'm trying to implement some javascript so that the page does not have to refresh every time someone votes on a comment, so that the vote goes through.
This is what I have in my comments controller:
def upvote_post_comment
#post = Post.find(params[:post_id])
#comment = #post.comments.find(params[:id])
#comment.liked_by current_user
respond_to do |format|
format.html {redirect_to :back }
format.json { render json: { count: #comment.votes.size } }
end
end
And this is what I have in my view:
<% if user_signed_in? && current_user != comment.user && !(current_user.voted_for? comment) %>
<%= link_to image_tag(‘vote.png'), like_post_comment_path(#post, comment), method: :put, class: 'vote', remote: true, data: { type: :json } %>
<a><%= "#{comment.votes.size}"%></a>
<script>
$('.vote')
.on('ajax:send', function () { $(this).addClass('loading'); })
.on('ajax:complete', function () { $(this).removeClass('loading'); })
.on('ajax:error', function(e, xhr, status, error) { console.log(status); console.log(error); })
.on('ajax:success', function (e, data, status, xhr) { $(this).html(data.count); });
</script>
<% elsif user_signed_in? && (current_user = comment.user) %>
<%= image_tag(‘voted.png')%><a><%= "#{comment.votes.size}"%></a>
<% else %>
<%= image_tag(‘voted.png')%><a><%= "#{comment.votes.size}"%></a>
<% end %>
When I click on the vote.png icon, the icon disappears and the vote count shows up instead of it. How do I get the vote.png icon to change to the voted.png icon and the vote count to update within the designated placeholder?
<a><%= "#{comment.votes.size}"%></a>
The rendered HTML:
<p>
<p>comment test</p>
<a class="vote" data-method="put" data-remote="true" data-type="json" href="/1/comments/8/like" rel="nofollow">
<img src="/assets/vote.png" />
<span>0</span>
</a>
</p>
The simple answer:
You're updating your entire .vote element (including your image). You need to differentiate between .vote img and .vote span:
<%= link_to like_post_comment_path(#post, comment), method: :put, class: 'vote', remote: true, data: { type: :json } do %>
<%= image_tag(‘vote.png') %>
<%= content_tag :span, comment.votes.size %>
<% end %>
.on('ajax:success', function (e, data, status, xhr) { $(this).find('span').html(data.count); }
Re-Factored Answer
#app/assets/javascripts/application.js
$(document)
.on('ajax:send', '.vote', function () { $(this).addClass('loading'); })
.on('ajax:complete', '.vote', function () { $(this).removeClass('loading'); })
.on('ajax:error', '.vote', function(e, xhr, status, error) { console.log(status); console.log(error); })
.on('ajax:success', '.vote', function (e, data, status, xhr) { $(this).find("span").html(data.count); });
#view
<% if user_signed_in? && current_user != comment.user %>
<% if !(current_user.voted_for? comment) %>
<%= link_to like_post_comment_path(#post, comment), method: :put, class: 'vote', remote: true, data: { type: :json } do %>
<%= image_tag('vote.png') %>
<%= content_tag :span, comment.votes.size %>
<% end %>
<% else %>
<%= image_tag('voted.png')%><a><%= "#{comment.votes.size}"%></a>
<% end %>
<% end %>
You're replacing your vote.png on your on success call.
.on('ajax:success', function (e, data, status, xhr) { $(this).html(data.count); });
Replace "this" with the class of your comment.votes.size.

Categories

Resources