Javascript不是在脚本中呈现而是作为html呈现

时间:2010-08-29 13:56:13

标签: ruby-on-rails ajax mime render unobtrusive-javascript

这个问题是我先前问过的yesterday,它涉及将模型的创建/编辑功能移动到其索引页面上。我遇到的最后一个问题是,当我去删除模型时,我有一些应该运行的javascript重新加载模型列表以反映数据库中的更改。这个js被渲染为html。以下是我认为相关的代码:

在我的控制器中:

def destroy
  @post = Post.find(params[:id])
  @post.destroy
  flash[:notice] = "Successfully destroyed post."
  @posts = Post.all
  respond_to do |format|
    format.js {render :content_type => 'text/javascript' }
  end
end

我的帖子部分列表(其中包含我所指的销毁链接):

<table>
  <tr>
    <th>Title</th>
    <th>Content</th>
  </tr>
  <% for post in @posts %>
    <tr>
      <td><%= post.title %></td>
      <td><%= post.content %></td>
      <td><%= link_to "Edit", edit_post_path(post), :class => "edit" %></td>
      <!--
        This is the problem here... I can't seem to get it to work.
        It DOES delete the record, but then gets redirected to /posts/:id
        and displays my javascript as html.
      -->
      <td><%= link_to "Destroy", post , :confirm => 'Are you sure?', :method => :delete, :class => 'destroy' %></td>
    </tr>
  <% end %>
</table>

destroy.js.erb:

$("#post_errors").hide(300);
$("#flash_notice").html("<%= escape_javascript(flash[:notice])%>");
$("#flash_notice").show(300);
$("#posts_list").html("<%= escape_javascript( render(:partial => "posts") ) %>");

在页面加载时加载的javscript:

// Setting up the ajax requests for the forms/links.
$(document).ready(function() {
  $("#new_post").submitWithAjax();
  $("a.edit").each(function(){
    $(this).getWithAjax();
  });
  $("a.destroy").each(function(){
    $(this).postWithAjax();
  });
});

// Setting up ajax for sending javascript requests
jQuery.ajaxSetup({
  'beforeSend': function(xhr) {xhr.setRequestHeader("Accept", "text/javascript")}
});

jQuery.fn.submitWithAjax = function() {
  this.submit(function() {
    $.post(this.action, $(this).serialize(), null, "script");
    return false;
  });
  return this;
};

jQuery.fn.getWithAjax = function() {
  this.click(function() {
    $.get(this.href, null, null, "script");
    return false;
  });
  return this;
};

jQuery.fn.postWithAjax = function() {
  this.click(function() {
    $.post(this.href, null, null, "script");
    return false;
  });
  return this;
};

要查看我正在使用的所有代码,请查看我发布的其他问题,但我认为这足以看出我做错了什么。此外,当我运行chrome并单击destroy链接时,chrome会向我显示一个javascript警告:

Resource interpreted as document but transferred with MIME type text/javascript.

3 个答案:

答案 0 :(得分:3)

此问题的答案是,在请求标头中,Accept标头应设置为text/javascript或类似内容。这应该在这个js函数中完成:

jQuery.ajaxSetup({
  'beforeSend': function(xhr) {xhr.setRequestHeader("Accept", "text/javascript")}
});

对于POST请求,这种情况发生得很好,但是当我想要执行PUT请求或DELETE请求时,这不会以某种方式被调用或者它正在被调用忽略。我意识到需要引用jQuery / AJAX的rails.js插件,因为我使用的是HTML5。

为了使编辑和删除工作,我对link_to方法执行了此操作:

link_to "Edit", edit_post_path(post), "data-remote" => 'true', 'data-method' => 'get'      
link_to "Destroy", post , "data-confirm" => 'Are you sure?', "data-remote" => 'true', "data-method" => 'delete'

将这些html属性添加到我的锚标记允许rails.js确定我想用它们做一些ajax请求。他们之后就开始工作了。

答案 1 :(得分:0)

手动设置MIME类型可能是问题的根源。 JavaScript通常以application/x-javascript而非text/javascript发送,因此最好将其留给渲染引擎为您设置。

对于像这样的简单案例,使用.js.rjs可能会更好,因为它会使您的JavaScript在很大程度上独立于框架。

答案 2 :(得分:0)

我遇到了同样的问题。 解决方案是使用此处提供的脚本:http://www.danhawkins.me.uk/2010/05/unobtrusive-javascript-with-rails-2-3-5/
在按照描述使用它之后,你所要做的就是使用类似于这个代码的东西:

 <%= link_to "Destroy", post , :confirm => 'Are you sure?', :class => 'remote destroy' %>

对我而言,这就像一个魅力。