Rails: fields_for con indice?


102

Esiste un metodo (o un modo per ottenere funzionalità simili) per eseguire una fields_for_with_index?

Esempio:

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

Quel parziale in fase di rendering deve sapere quale sia l'indice corrente nel fields_forciclo.


3
Farei + uno questo in core rail ... Continua a imbattermi in un caso anche per questo.
Nathan Bertram

Risposte:


92

Questo sarebbe effettivamente un approccio migliore, seguendo più da vicino la documentazione di Rails:

<% @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 %>

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

È anche possibile specificare l'istanza da utilizzare:

  <%= 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 %>

2
Vorrei che ci fosse qualcosa integrato in fields_for per questo, ma poiché non c'è la tua risposta mi ha salvato la giornata. Grazie.
Anders Kindberg

1
Puoi anche utilizzare l'opzione non documentata: child_index su fields_for, se hai bisogno di un maggiore controllo su quale indice viene visualizzato, in questo modo: fields_for (: projects, project, child_index: index)
Anders Kindberg

Questo è il collegamento Rails API funzionante api.rubyonrails.org/classes/ActionView/Helpers/…
Archonic

7
Gli utenti di Rails 4.0.2+ dovrebbero controllare la risposta di Ben poiché index è ora integrato nel builder.
notapatch

1
@Marco Tu sei il re signore. mi hai salvato giorno per giorno :-) Vorrei poterti regalare un paio di 10 + ......!

157

La risposta è abbastanza semplice poiché la soluzione è fornita all'interno di Rails. Puoi usare f.optionsparams. Quindi, all'interno del tuo rendering _some_form.html.erb,

L'indice è accessibile da:

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

Non devi fare nient'altro.


Aggiornamento: sembra che la mia risposta non fosse abbastanza chiara ...

File HTML originale:

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

Sottomodulo renderizzato:

<!-- _some_form.html.erb -->
<%= f.options[:child_index] %>

10
non funziona qui. potete fornire link alla documentazione di rails?
Lucas Renan

2
@LucasRenan & @graphmeter - Si prega di leggere di nuovo la domanda, è necessario chiamare <%= f.options[:child_index] %>il sotto-modulo renderizzato (in questo caso: _some_form.html.erb), non nell'originale builder. Risposta aggiornata per maggiori chiarimenti.
Sheharyar

1
Strano, capisco nilper quello
bcackerman

1
@Sheharyar Funziona in Rails 4. Ma questo mi dà valori come "1450681048049,1450681050158,1450681056830,1450681141951,1450681219262". Ma ho bisogno di un indice in questa forma "1,2,3,4,5". Cosa dovrei fare?
vidal

2
Brillante. Questa dovrebbe essere assolutamente la risposta accettata.
jeffdill2


17

Per Rails 4+

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

Supporto per Monkey Patch per Rails 3

Per iniziare f.indexa lavorare in Rails 3 è necessario aggiungere una patch scimmia agli inizializzatori dei progetti a cui aggiungere questa funzionalitàfields_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

Risposta migliore finora, l'uso .indexè molto più pulito.
Lucas Andrade

7

Checkout Rendering di una raccolta di parziali . Se il tuo requisito è che un modello debba iterare su un array e renderizzare un sottomodello per ciascuno degli elementi.

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

Questo renderà "_children.erb" e passerà la variabile locale 'children' al modello per la visualizzazione. Un contatore di iterazioni verrà automaticamente reso disponibile al modello con un nome del modulo partial_name_counter. Nel caso dell'esempio sopra, il modello verrebbe alimentato children_counter.

Spero che questo ti aiuti.


Ottengo "variabile locale non definita o metodo 'question_comment_form_counter'" quando lo faccio. question_comment_formessendo il mio nome parziale ...
Shpigford

Fai la domanda has_many: commenti e come stai creando commenti, ad esempio nella tua nuova azione o modifica delle domande prova a fare 1.times {question.comments.build}
Syed Aslam

5

Non riesco a vedere un modo decente per farlo attraverso i modi forniti da Rails, almeno non in -v3.2.14

@Sheharyar Naseer fa riferimento all'hash delle opzioni che può essere utilizzato per risolvere il problema, ma non per quanto posso vedere nel modo in cui sembra suggerire.

Ho fatto questo =>

<%= 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 %>

o

<%= 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 %>

Nel mio caso g.object_namerestituisce una stringa come questa "gallery_set[blog_posts_attributes][2]" per il terzo campo renderizzato, quindi abbino semplicemente l'indice in quella stringa e lo uso.


In realtà un modo più freddo (e forse più pulito?) Per farlo è passare un lambda e chiamarlo per incrementare.

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

E nella 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 %>

1

So che è un po 'tardi, ma recentemente ho dovuto farlo puoi ottenere l'indice dei campi_per in questo modo

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

Spero che questo aiuti :)


1

Aggiunto 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 %>

Questa è la nuova migliore risposta.
genkilabs

Qualcun altro ottiene campi duplicati con questo?

0

Se vuoi avere il controllo sugli indici, controlla l' indexopzione

<%= 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 =>

Questo produrrà

<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 il modulo viene inviato, params includerà qualcosa di simile

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

Ho dovuto usare l'opzione index per far funzionare i miei multi-dropdown. In bocca al lupo.

Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.