【问题标题】:Rails 4: update model attribute with AJAX callRails 4:使用 AJAX 调用更新模型属性
【发布时间】:2015-11-20 18:23:57
【问题描述】:

这是Rails 4: custom action to update only one param with AJAXRails 4: select DOM element with dynamically generated id 的后续问题。

我无法从我得到的好答案中弄清楚如何让事情发挥作用,所以我想从头开始。

在我的 Rails 4 应用程序中,我有一个 Calendar 和一个 Post 模型,设置了浅路径:

resources :calendars do
  resources :posts, shallow: true
end

一个日历has_many 帖子和一个帖子belong_to 一个日历。

在日历show.html.erb 视图中,我显示属于某个日历的所有帖子。

从此日历show.html.erb 视图中,我需要使用 AJAX 调用更新帖子的 :approval 参数,以便仅刷新与此特定帖子的 :approval 相关的 DOM 元素,而不是整个页面.

这是日历show.html.erb 中的代码部分,它允许我更新帖子的:approval

<tr id="post_row_<%= post.id%>"> 
  [...]  # Truncated for brivety 
  <td class="cell_content_center post_approval_section">
    <% if post.approval == "ok" %>
      <span class="ok_green">
    <% else %>
      <span class="approval_blue" %>
    <% end %>
      <%= link_to post_path(:id => post.id, "post[approval]" => "ok", approval_update: true), remote: true, :method => :patch do %>
        <span class="glyphicon glyphicon-ok" data-toggle="tooltip" data-placement="left" title="Approve Post"></span>
      <% end %>
    </span><br/>
    <% if post.approval == "edit" %>
      <span class="edit_yellow">
    <% else %>
      <span class="approval_blue" %>
    <% end %>
      <%= link_to post_path(:id => post.id, "post[approval]" => "edit", approval_update: true), remote: true, :method => :patch do %>
      <span class="glyphicon glyphicon-repeat" data-toggle="tooltip" data-placement="left" title="Require Edits"></span>
    <% end %>
    </span><br/>
    <% if post.approval == "remove" %>
      <span class="remove_red">
    <% else %>
      <span class="approval_blue" %>
    <% end %>
      <%= link_to post_path(:id => post.id, "post[approval]" => "remove", approval_update: true), remote: true, :method => :patch do %>
      <span class="glyphicon glyphicon-remove" data-toggle="tooltip" data-placement="left" title="To Be Deleted"></span>
    <% end %>
    </span>
  </td>
  [...]  # Truncated for brivety
</tr>

然后,在posts_controller.rb

def update
    if params["approval_update"]
      respond_to do |format|
        if @post.update(post_params)
          format.js { render :action => "update_post_approval" }
        else
          format.html { redirect_to :back }
        end
      end
    else
      respond_to do |format|
        if @post.update(post_params)
          format.html { redirect_to post_path(@post) }
          format.json { render :show, status: :ok, location: @post }
          format.js
        else
          format.html { render :edit }
          format.json { render json: @post.errors, status: :unprocessable_entity }
        end
      end
    end
  end

  def update_post_approval
    respond_to do |format|
      format.js
    end
  end

注意:在这里,我不能使用默认的 Posts#Update 操作,因为它已被另一个 AJAX 调用用于更新帖子中的 :approval 参数show.html.erb 视图,这就是我们渲染 update_post_approval 自定义操作的原因。

然后,我有以下app/views/posts/update_post_approval.js.erb 文件:

$("tr#post_row_#{post.id} > td.post_approval_section").html('<%= j render(partial: "calendars/post_approval") %>');

最后,我有以下calendars/_post_approval.html.erb partial:

<td class="cell_content_center post_approval_section">
  <% if post.approval == "ok" %>
    <span class="ok_green">
  <% else %>
    <span class="approval_blue" %>
  <% end %>
    <%= link_to post_path(:id => post.id, "post[approval]" => "ok", approval_update: true), remote: true, :method => :patch do %>
      <span class="glyphicon glyphicon-ok" data-toggle="tooltip" data-placement="left" title="Approve Post"></span>
    <% end %>
  </span><br/>
  <% if post.approval == "edit" %>
    <span class="edit_yellow">
  <% else %>
    <span class="approval_blue" %>
  <% end %>
    <%= link_to post_path(:id => post.id, "post[approval]" => "edit", approval_update: true), remote: true, :method => :patch do %>
    <span class="glyphicon glyphicon-repeat" data-toggle="tooltip" data-placement="left" title="Require Edits"></span>
  <% end %>
  </span><br/>
  <% if post.approval == "remove" %>
    <span class="remove_red">
  <% else %>
    <span class="approval_blue" %>
  <% end %>
    <%= link_to post_path(:id => post.id, "post[approval]" => "remove", approval_update: true), remote: true, :method => :patch do %>
    <span class="glyphicon glyphicon-remove" data-toggle="tooltip" data-placement="left" title="To Be Deleted"></span>
  <% end %>
  </span>
</td>

当我单击&lt;td class="cell_content_center post_approval_section"&gt; 部分中的三个链接之一来更新帖子:approval 参数时,参数实际上已更新,正如我们在服务器日志中看到的那样:

Started PATCH "/posts/38?approval_update=true&post%5Bapproval%5D=ok" for ::1 at 2015-11-20 08:03:50 -0800
Processing by PostsController#update as JS
  Parameters: {"approval_update"=>"true", "post"=>{"approval"=>"ok"}, "id"=>"38"}
  User Load (1.0ms)  SELECT  "users".* FROM "users" WHERE "users"."id" = $1  ORDER BY "users"."id" ASC LIMIT 1  [["id", 1]]
  Post Load (0.7ms)  SELECT  "posts".* FROM "posts" WHERE "posts"."id" = $1 LIMIT 1  [["id", 38]]
   (0.6ms)  BEGIN
  SQL (1.1ms)  UPDATE "posts" SET "approval" = $1, "updated_at" = $2 WHERE "posts"."id" = $3  [["approval", "ok"], ["updated_at", "2015-11-20 16:03:50.888587"], ["id", 38]]
  SQL (1.3ms)  INSERT INTO "versions" ("event", "object", "whodunnit", "created_at", "object_changes", "item_id", "item_type") VALUES ($1, $2, $3, $4, $5, $6, $7) RETURNING "id"  [["event", "update"], ["object", "---\nid: 38\ncalendar_id: 6\ndate: 2015-11-17 22:09:00.000000000 Z\nsubject: Subject\nformat: Link\ncopy: https://www.wid.la/\ncreated_at: 2015-11-17 22:09:28.713416000 Z\nupdated_at: 2015-11-20 16:03:20.613597000 Z\nimage_file_name: \nimage_content_type: \nimage_file_size: \nimage_updated_at: \nshort_copy: ''\nscore: \nfacebook: true\ntwitter: false\ninstagram: false\npinterest: false\ngoogle: false\nlinkedin: false\ntumblr: \nsnapchat: \napproval: remove\n"], ["whodunnit", "1"], ["created_at", "2015-11-20 16:03:50.888587"], ["object_changes", "---\napproval:\n- remove\n- ok\nupdated_at:\n- 2015-11-20 16:03:20.613597000 Z\n- 2015-11-20 16:03:50.888587000 Z\n"], ["item_id", 38], ["item_type", "Post"]]
   (1.6ms)  COMMIT
  Rendered calendars/_post_approval.html.erb (0.1ms)
  Rendered posts/update_post_approval.js.erb (1.5ms)
Completed 200 OK in 27ms (Views: 9.9ms | ActiveRecord: 6.3ms)

但是,除非我手动刷新页面,否则视图不会发生任何变化。

我现在已经在这个问题上花费了几个小时,但无法确定问题是由 Ruby 还是 JS 代码引起的。

如有必要,我很乐意分享更多代码。

知道我在这里做错了什么吗?

【问题讨论】:

  • 你可以尝试给出一个打印语句来打印 app/views/posts/update_post_approval.js.erb 文件中的帖子 ID,如果它根据日志得到更新,那表示问题出在 js 文件上。对于js文件中非常微小的错误,js文件中编写的函数将不会执行。

标签: javascript jquery ruby-on-rails ajax ruby-on-rails-4


【解决方案1】:

错误就在这里:

$("tr#post_row_#{post.id} > td.post_approval_section").html('');

更新 应该是:

$("tr#post_row_<%= @post.id %> > td.post_approval_section").html('<%= j render(partial: "calendars/post_approval", locals: {post: @post}) %>');

【讨论】:

  • 感谢您的回答。不幸的是,这似乎并不能解决问题。它会导致服务器日志中出现问题:NameError - undefined local variable or method post' for #:0x007fab69e1f160>: app/views/posts/update_post_approval.js.erb:1:in _app_views_posts_update_post_approval_js_erb__1803217690360989333_70187072192720'
  • 是的,我有这个疑问,请将帖子更改为@post,并在部分中传递一个本地参数:$("tr#post_row_&lt;%= post.id %&gt; &gt; td.post_approval_section").html('&lt;%= j render(partial: "calendars/post_approval", locals: {post: @post}) %&gt;');
  • 先生,您是摇滚明星。它工作得很好,非常感谢。
  • 很高兴它解决了你的问题,我自己也遇到过很多次了。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2012-01-28
  • 2017-04-18
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多