Динамически добавлять поля в рельсы без вложенных атрибутов
Я нахожусь на ранних этапах создания приложения, и просто устанавливаю базовый код. Вот текущий код...
приложение/просмотров/карты/front.html.erb
<%= form_for(front_of_card_path) do |f| %>
<%= f.fields_for :competency_templates do |builder| %>
<%= render 'add_fields', f: builder %>
<% end %>
<%= link_to_add_fields "Add New Tag", f, :skill %>
<% end %>
маршруты
controller :cards do
get '/front', action: 'front', as: 'front_of_card'
post '/save', action: 'create', as: 'save_card'
get '/my_contact_info', action: 'back', as: 'back_of_card'
put '/save', action: 'update', as: 'save_card'
get '/my_card', action: 'show', as: 'card'
end
контроллер
def create
@skill= Skill.new(params[:skill])
@tag = Tag.new(params[:tag])
@tag.save
@skill.tag_id = @tag.id
@skill.save
redirect_to front_of_card_path, notice: 'Skill was successfully created.'
#get user/session
#save skills & tags
end
cards.js.coffee
jQuery ->
$('form').on 'click', '.remove_fields', (event) ->
$(this).prev('input[type=hidden]').val('1')
$(this).closest('fieldset').hide()
event.preventDefault()
$('form').on 'click', '.add_fields', (event) ->
time = new Date().getTime()
regexp = new RegExp($(this).data('id'), 'g')
$(this).before($(this).data('fields').replace(regexp, time))
event.preventDefault()
app_helper
module ApplicationHelper
def link_to_add_fields(name, f, association)
new_object = f.object.send(association).klass.new
id = new_object.object_id
fields = f.fields_for(association, new_object, child_index: id) do |builder|
render(association.to_s.singularize + "_fields", f: builder)
end
link_to(name, '#', class: "add_fields", data: {id: id, fields: fields.gsub("\n", "")})
end
end
Итак, сейчас этот код дает мне два текстовых поля. Один для имени тега, а другой для веса тега, а контроллер вставляет все в БД. Я хотел бы использовать некоторый javascript для динамического добавления как можно большего количества этих полей тегов/вес, как мне нравится. Все, что я нашел, похоже, сосредоточено на вложенных атрибутах. Любые идеи оценили.
Update
Добавлен больше кода для этого. Проблема, с которой я сталкиваюсь, - это 3-я переменная, которую я передаю в этой строке...
<%= link_to_add_fields "Add New Tag", f, :skill %>
Это не нравится ': skill', но я не уверен, что я должен здесь проходить.
Ответы
Ответ 1
Итак, вот что я придумал... вот мои две модели...
class Skill < ActiveRecord::Base
belongs_to :tag
attr_accessible :tag_id, :weight
end
class Tag < ActiveRecord::Base
has_many :skills
attr_accessible :name
end
Я вызываю частичное из app/views/skills/_form.html.erb и используя тег js для добавления новых полей. Также обратите внимание, что я повторно рендеринг частичного, а затем скрываю его в последнем теге div.
<div id="skillSet">
<%= render partial: "skills_form" %>
</div>
<a href="javascript:;" id="addNewTag">Add New Tag</a>
<div class="actions">
<%= f.submit %>
</div>
<% end %>
<div class="hide" id="new_skills_form">
<%= render partial: "skills_form", locals: {skill: false} %>
</div>
Частичное довольно просто. Все, что я делаю здесь, это сохранение значений в массиве...
<div class="skillsForm">
<%= label_tag 'tag' %>
<%= text_field_tag 'tags[]' %>
<%= label_tag 'weight' %>
<%= text_field_tag 'weights[]' %>
</div>
... вот javascript... реальный прямо, просто скажите, когда нажата кнопка #addNewTag, приложите #new_skills_form к #skillSet
$(document).ready(function(){
$("#addNewTag").click(function(){
$("#skillSet").append($("#new_skills_form").html());
});
});
... и, наконец, действие контроллера деконструирует массивы и сохраняет их...
def create
@skill = Skill.new(params[:skill])
tags = params[:tags]
weights = params[:weights]
tags.each_with_index do |tag, index|
tag = Tag.create :name => tag
Skill.create :tag_id => tag.id, :weight => weights[index]
end
end