Rails: fields_for com índice?

102

Existe um método (ou maneira de obter funcionalidade semelhante) para fazer um fields_for_with_index?

Exemplo:

<% f.fields_for_with_index :questions do |builder, index| %>  
  <%= render 'some_form', :f => builder, :i => index %>
<% end %>

Esse parcial sendo renderizado precisa saber qual é o índice atual no fields_forloop.

Shpigford
fonte
3
Eu marcaria um com este em trilhos centrais ... Continuo encontrando um caso para este também.
Nathan Bertram

Respostas:

92

Na verdade, essa seria uma abordagem melhor, seguindo a documentação do Rails mais de perto:

<% @questions.each.with_index do |question,index| %>
    <% f.fields_for :questions, question do |fq| %>  
        # here you have both the 'question' object and the current 'index'
    <% end %>
<% end %>

De: http://railsapi.com/doc/rails-v3.0.4/classes/ActionView/Helpers/FormHelper.html#M006456

Também é possível especificar a instância a ser usada:

  <%= form_for @person do |person_form| %>
    ...
    <% @person.projects.each do |project| %>
      <% if project.active? %>
        <%= person_form.fields_for :projects, project do |project_fields| %>
          Name: <%= project_fields.text_field :name %>
        <% end %>
      <% end %>
    <% end %>
  <% end %>
Marco Lazzeri
fonte
2
Eu gostaria que houvesse algo construído em fields_for para isso, mas como não houve sua resposta salvou meu dia. Obrigado.
Anders Kindberg
1
Você também pode usar a opção não documentada: child_index em fields_for, se precisar de mais controle sobre qual índice é renderizado, como este: fields_for (: projects, project, child_index: index)
Anders Kindberg
Este é o link funcional da API do Rails api.rubyonrails.org/classes/ActionView/Helpers/…
Archonic
7
Os usuários do Rails 4.0.2+ devem verificar a resposta de Ben, pois o índice agora está embutido no construtor.
notapatch de
1
@Marco Você é o rei, senhor. você salvou meu dia a dia. :-) Queria poder te dar mais de 10 ......!
157

A resposta é bastante simples, pois a solução é fornecida pelo Rails. Você pode usar f.optionsparams. Então, dentro do seu renderizado _some_form.html.erb,

O índice pode ser acessado por:

<%= f.options[:child_index] %>

Você não precisa fazer mais nada.


Atualização: Parece que minha resposta não foi clara o suficiente ...

Arquivo HTML original:

<!-- Main ERB File -->
<% f.fields_for :questions do |builder| %>  
  <%= render 'some_form', :f => builder %>
<% end %>

Sub-formulário renderizado:

<!-- _some_form.html.erb -->
<%= f.options[:child_index] %>
Sheharyar
fonte
10
não está funcionando aqui. você pode fornecer o link de documentação do Rails?
Lucas Renan
2
@LucasRenan & @graphmeter - Por favor, leia a pergunta novamente, você precisa chamar <%= f.options[:child_index] %>seu sub-formulário renderizado (neste caso: _some_form.html.erb), não no original builder. Resposta atualizada para mais esclarecimentos.
Sheharyar
1
Estranho, eu acho nilisso
bcackerman
1
@Sheharyar Isso está funcionando no Rails 4. Mas está me dando valores como '1450681048049,1450681050158,1450681056830,1450681141951,1450681219262'. Mas preciso do índice neste formato '1,2,3,4,5'. O que devo fazer?
vidal
2
Brilhante. Esta deve ser absolutamente a resposta aceita.
jeffdill2
100

A partir do Rails 4.0.2, um índice agora está incluído no objeto FormBuilder:

https://api.rubyonrails.org/classes/ActionView/Helpers/FormBuilder.html#method-i-fields_for

Por exemplo:

<%= form_for @person do |person_form| %>
  ...
  <%= person_form.fields_for :projects do |project_fields| %>
    Project #<%= project_fields.index %>
  ...
  <% end %>
  ...
<% end %>
Ben
fonte
Isso funciona e é menos para escrever do que project_fields.options[:child_index]. Então eu gosto mais assim!
Casey,
17

Para Rails 4+

<%= form_for @person do |person_form| %>
  <%= person_form.fields_for :projects do |project_fields| %>
    <%= project_fields.index %>
  <% end %>
<% end %>

Suporte do Monkey Patch para Rails 3

Para começar f.indexa trabalhar no Rails 3 você precisa adicionar um monkey patch aos inicializadores de seus projetos para adicionar esta funcionalidade afields_for

# config/initializers/fields_for_index_patch.rb

module ActionView
  module Helpers
    class FormBuilder

      def index
        @options[:index] || @options[:child_index]
      end

      def fields_for(record_name, record_object = nil, fields_options = {}, &block)
        fields_options, record_object = record_object, nil if record_object.is_a?(Hash) && record_object.extractable_options?
        fields_options[:builder] ||= options[:builder]
        fields_options[:parent_builder] = self
        fields_options[:namespace] = options[:namespace]

        case record_name
          when String, Symbol
            if nested_attributes_association?(record_name)
              return fields_for_with_nested_attributes(record_name, record_object, fields_options, block)
            end
          else
            record_object = record_name.is_a?(Array) ? record_name.last : record_name
            record_name   = ActiveModel::Naming.param_key(record_object)
        end

        index = if options.has_key?(:index)
                  options[:index]
                elsif defined?(@auto_index)
                  self.object_name = @object_name.to_s.sub(/\[\]$/,"")
                  @auto_index
                end

        record_name = index ? "#{object_name}[#{index}][#{record_name}]" : "#{object_name}[#{record_name}]"
        fields_options[:child_index] = index

        @template.fields_for(record_name, record_object, fields_options, &block)
      end

      def fields_for_with_nested_attributes(association_name, association, options, block)
        name = "#{object_name}[#{association_name}_attributes]"
        association = convert_to_model(association)

        if association.respond_to?(:persisted?)
          association = [association] if @object.send(association_name).is_a?(Array)
        elsif !association.respond_to?(:to_ary)
          association = @object.send(association_name)
        end

        if association.respond_to?(:to_ary)
          explicit_child_index = options[:child_index]
          output = ActiveSupport::SafeBuffer.new
          association.each do |child|
            options[:child_index] = nested_child_index(name) unless explicit_child_index
            output << fields_for_nested_model("#{name}[#{options[:child_index]}]", child, options, block)
          end
          output
        elsif association
          fields_for_nested_model(name, association, options, block)
        end
      end

    end
  end
end
Weston Ganger
fonte
Melhor resposta até agora, o uso .indexé muito mais limpo.
Lucas Andrade
7

Checkout Renderizando uma coleção de parciais . Se o seu requisito é que um modelo precise iterar sobre um array e renderizar um submodelo para cada um dos elementos.

<%= f.fields_for @parent.children do |children_form| %>
  <%= render :partial => 'children', :collection => @parent.children, 
      :locals => { :f => children_form } %>
<% end %>

Isso irá renderizar “_children.erb“ e passar a variável local 'children' para o modelo para exibição. Um contador de iteração será automaticamente disponibilizado para o modelo com um nome do formulário partial_name_counter. No caso do exemplo acima, o modelo seria alimentado children_counter.

Espero que isto ajude.

Syed Aslam
fonte
Recebo "variável local indefinida ou método 'question_comment_form_counter'" ao fazer isso. question_comment_formsendo meu nome parcial ...
Shpigford
Faça question has_many: comments e como você está construindo comentários, por exemplo em sua ação nova ou de edição das perguntas, tente fazer 1.vezes {question.comments.build}
Syed Aslam
5

Não consigo ver uma maneira decente de fazer isso através das formas fornecidas pelo Rails, pelo menos não em -v3.2.14

@Sheharyar Naseer faz referência às opções de hash que podem ser usadas para resolver o problema, mas não tanto quanto posso ver da maneira que ele parece sugerir.

Eu fiz isso =>

<%= f.fields_for :blog_posts, {:index => 0} do |g| %>
  <%= g.label :gallery_sets_id, "Position #{g.options[:index]}" %>
  <%= g.select :gallery_sets_id, @posts.collect  { |p| [p.title, p.id] } %>
  <%# g.options[:index] += 1  %>
<% end %>

ou

<%= f.fields_for :blog_posts do |g| %>
  <%= g.label :gallery_sets_id, "Position #{g.object_name.match(/(\d+)]/)[1]}" %>
  <%= g.select :gallery_sets_id, @posts.collect  { |p| [p.title, p.id] } %>
<% end %>

No meu caso, g.object_nameretorna uma string como esta "gallery_set[blog_posts_attributes][2]" para o terceiro campo renderizado, então eu apenas combino o índice nessa string e o uso.


Na verdade, uma maneira mais legal (e talvez mais limpa?) De fazer isso é passar um lambda e chamá-lo para incrementar.

# /controller.rb
index = 0
@incrementer = -> { index += 1}

E na vista

<%= f.fields_for :blog_posts do |g| %>
  <%= g.label :gallery_sets_id, "Position #{@incrementer.call}" %>
  <%= g.select :gallery_sets_id, @posts.collect  { |p| [p.title, p.id] } %>
<% end %>
iNulty
fonte
1

Eu sei que é um pouco tarde, mas recentemente tive que fazer isso você pode obter o índice de fields_for assim

<% f.fields_for :questions do |builder| %>
  <%= render 'some_form', :f => builder, :i => builder.options[:child_index] %>
<% end %>

Eu espero que isso ajude :)

MZaragoza
fonte
1

Adicionado a fields_for child_index: 0

<%= form_for @person do |person_form| %>
  <%= person_form.fields_for :projects, child_index: 0 do |project_fields| %>
    <%= project_fields.index %>
  <% end %>
<% end %>
gilcierweb
fonte
Esta é a nova melhor resposta.
genkilabs
Alguém mais obtém campos duplicados com isso?
0

Se você deseja ter controle sobre os índices verifique a indexopção

<%= f.fields_for :other_things_attributes, @thing.other_things.build do |ff| %>
  <%= ff.select :days, ['Mon', 'Tues', 'Wed'], index: 2 %>
  <%= ff.hidden_field :special_attribute, 24, index: "boi" %>
<%= end =>

Isso vai produzir

<select name="thing[other_things_attributes][2][days]" id="thing_other_things_attributes_7_days">
  <option value="Mon">Mon</option>
  <option value="Tues">Tues</option>
  <option value="Wed">Wed</option>
</select>
<input type="hidden" value="24" name="thing[other_things_attributes][boi][special_attribute]" id="thing_other_things_attributes_boi_special_attribute">

Se o formulário for enviado, os parâmetros incluirão algo como

{
  "thing" => {
  "other_things_attributes" => {
    "2" => {
      "days" => "Mon"
    },
    "boi" => {
      "special_attribute" => "24"
    }
  }
}

Tive de usar a opção de índice para fazer meus vários menus suspensos funcionarem. Boa sorte.

Cruz Nunez
fonte