1

我只是在学习 Rails,如果有任何帮助,我将不胜感激。我已经看了大概 10 个教程:remote => trueform_for它们似乎都互相复制(同样令人困惑)。

本教程就是其中之一:http ://tech.thereq.com/post/18910961502/rails-3-using-form-remote-true-with-jquery-ujs

我的网络应用程序几乎都在一个页面上,我的“提交表单”在一个弹出式灯箱内。这意味着 form_for 代码实际上在我的 index.html.erb 中。这也意味着创建了一个新的@playlist 实例变量index(而不是让它在new

我的最终目标是

if @playlist.save返回true,我想将用户重定向到root_pathWITH a :notice。无需使用成功消息或类似内容更新表单。我想在保存成功时进行硬重定向。

if @playlist.save返回false,我想渲染 create.js.erb 中的代码,这会将错误附加到灯箱表单中而不刷新页面。note很多教程似乎都想将整个表单重新渲染为 html,然后将内容替换到位。我宁愿(如果可以的话),只需将错误发送到表单并插入它们,我认为重新渲染和替换整个表单似乎比它必须的更复杂。

截至目前,我的代码有点混乱,尤其是:

  • respond_to块。这是关于这里实际发生的最大混淆点,我尝试了很多不同的代码,但没有任何效果
  • 我不知道 create.js.erb 是否正确
  • 我没有 create.html.erb - 我认为这是正确的,因为我只想重定向

我的表格的第一行

<%= form_for @playlist, :remote => true do |f| %>

我的整个播放列表控制器

class PlaylistsController < ApplicationController

  before_filter :load_genres, :only =>[:index, :user]
  before_filter :new_playlist, :only => [:index, :new, :create]

  def index
    @playlists = Playlist.order('created_at DESC').page(params[:page]).per(30)
  end

  def new
  end

  def create
    respond_to do |format|
      if @playlist.save
        # ???
        format.html { redirect_to root_path, :notice => "Playlist submitted" }
      else
        # ???
        format.js { render :js => @playlist.errors, :status => :unprocessable_entity }
      end
    end
  end

  def user
    @playlists = Playlist.order('created_at DESC').where(:username => params[:username]).page(params[:page]).per(30)
  end

  def listen
    playlist = Playlist.find(params[:playlist_id])
    playlist.update_attribute(:listens, playlist.listens + 1)
    render :nothing => true
  end

  private

  def load_genres
    @genres = Genre.order(:name)
  end

  def new_playlist
    @playlist = Playlist.new(:title => params[:title], :url => params[:url], :description => params[:description])
  end

end

创建.js.erb

jQuery(function($) {
    alert('createjs called');
    $('#submit-form form').on('ajax:error', function(event, xhr, status, error){

        var responseObject = $.parseJSON(xhr.responseText), 
            errors = $('<ul />');

        $.each(responseObject, function(index, value){
            errors.append('<li>' + value + '</li>');
        })

        $(this).find('.submit-playlist-errors').html(errors);
    });
});
4

1 回答 1

3

flash如果我对您的理解正确,您有一个表格,如果表格已保存,您想要重定向并发出通知。如果表单无效,您希望显示错误。

您可以将表单(或任何请求)以不同的格式(通常是 html、json 和 js)发送到 rails 应用程序。这是在您提供的网址中指定的。

playlist_path(@playlist)
# => /playlists/1

playlist_path(@playlist, :format => "json")
# => /playlists/1.json

playlist_path(@playlist, :format => "html")
# => /playlists/1.html
# ... etc

respond_to该格式告诉 Rails要使用您的块中指定的响应类型。

现在,如果这很清楚,您必须决定使用哪种格式。

如果您指定js格式,响应将被视为script已执行。如果表单有效,您可以发送用户 javascript,将他们重定向到您想要的页面。如果表单无效,您可以通过修改 HTML 的方式来显示错误消息。不需要客户端代码。

respond_to do |format|
  if valid
    format.js { render "valid" }
  else
    format.js { render "invalid", :status => :unprocessable_entity }
  end
end

# valid.js.erb
window.location = <%= your_path %>

# invalid.js.erb
// whatever to display errors

当您使用时json,如果表单无效,您可以返回重定向 url 以重定向用户或错误。请注意,此解决方案需要在客户端使用 Javascript 来处理它。

# controller
respond_to do |format|
  if valid
    format.json { render :json => your_path }
  else
    format.json { render :json => @object.errors, :status => :unprocessable_entity }
  end
end

# client side javascript
$("form")
  .on("ajax:success", function(event, data, status, response) {
    window.location = data
  })
  .on("ajax:error", function(event, response, error) {
    // logic to handle errors
  })

您还可以使用html格式(通过将格式设置为html或根本不指定格式)。但是,如果您使用,则必须以与(具有客户端 javascript):remote => true相同的方式对待它。json如果您redirect_to用作响应,它将重定向请求,而不是您的用户。

我不确定在用户被重定向后是否显示 Flash 消息,但您可以通过以下方式尝试:

respond_to do |format|
  if valid
    format.html { 
      flash[:notice] = "Playlist submitted"
      render :json => your_path }
  else
    format.html { render :json => @object.errors, :status => :unprocessable_entity }
  end
end

# client side javascript
//...

希望这会让你的respond_to街区变干。

于 2012-09-24T17:16:03.787 回答