cocoon nested_form在edit.html.erb中添加现有对象上方的字段,而不是在下面

时间:2012-12-11 08:57:46

标签: jquery ruby-on-rails nested-forms cocoon-gem

我的new.html.erb工作正常。

但在我的edit.html.erb中,假设我有3个现有部门......

每当我点击添加部门链接(添加另一个字段)时,它就会继续在现有部门之上,而不是在下面。

不过,我的协会说,师有很多部门和部门属于师。

在我的控制器中:我的新方法和编辑方法

def new
    @division = Division.new
     @division.departments.build 
  end

 def edit
    @division = Division.find(params[:id])

  end

我需要cocoon js文件,所以我使用的是默认值而不是覆盖别的东西。

cocoon.js

(function($) {

  function replace_in_content(content, regexp_str, with_str) {
    reg_exp = new RegExp(regexp_str);
    content.replace(reg_exp, with_str);
  }

  function trigger_removal_callback(node) {
    node.parent().parent().trigger('removal-callback');
  }

  $('.add_fields').live('click', function(e) {
    e.preventDefault();
    var $this                 = $(this),
        assoc                 = $this.data('association'),
        assocs                = $this.data('associations'),
        content               = $this.data('template'),
        insertionMethod       = $this.data('association-insertion-method') || $this.data('association-insertion-position') || 'before';
        insertionNode         = $this.data('association-insertion-node'),
        insertionCallback     = $this.data('insertion-callback'),
        removalCallback       = $this.data('removal-callback'),
        regexp_braced         = new RegExp('\\[new_' + assoc + '\\]', 'g'),
        regexp_underscord     = new RegExp('_new_' + assoc + '_', 'g'),
        new_id                = new Date().getTime(),
        newcontent_braced     = '[' + new_id + ']',
        newcontent_underscord = '_' + new_id + '_',
        new_content           = content.replace(regexp_braced, '[' + new_id + ']');

    if (new_content == content) {
        regexp_braced     = new RegExp('\\[new_' + assocs + '\\]', 'g');
        regexp_underscord = new RegExp('_new_' + assocs + '_', 'g');
        new_content       = content.replace(regexp_braced, '[' + new_id + ']');
    }

    new_content = new_content.replace(regexp_underscord, newcontent_underscord);

    if (insertionNode){
      insertionNode = insertionNode == "this" ? $this : $(insertionNode);
    } else {
      insertionNode = $this.parent();
    }

    var contentNode = $(new_content);

    // allow any of the jquery dom manipulation methods (after, before, append, prepend, etc)
    // to be called on the node.  allows the insertion node to be the parent of the inserted
    // code and doesn't force it to be a sibling like after/before does. default: 'before'
    insertionNode[insertionMethod](contentNode);

    $this.parent().trigger('insertion-callback');
  });

  $('.remove_fields.dynamic').live('click', function(e) {
    var $this = $(this);
    trigger_removal_callback($this);
    e.preventDefault();
    $this.closest(".nested-fields").remove();
  });

  $('.remove_fields.existing').live('click', function(e) {
    var $this = $(this);
    trigger_removal_callback($this);
    e.preventDefault();
    $this.prev("input[type=hidden]").val("1");
    $this.closest(".nested-fields").hide();
  });

})(jQuery);

在我的_form.html.erb

<div class="new-div-box">
<%= simple_form_for [:admin, @division]  do |f| %>
  <%= f.input :name, :label => "Division Name"%>

<div>
<div>

  <%= f.simple_fields_for :departments do |department| %>
  <%= render 'department_fields', :f => department %>

  <% end %>
</div>
  <%=link_to_add_association "Add Department", f, :departments,class: "btn btn-inverse" %>
</div>

  <%= f.submit  :class=>"btn btn-primary submit_btn_division" %>

<% end %>

</div>

我呈现的是_department_fields.html.erb

<div class="nested-fields">
  <%= f.input :name, :label => "Department Name"  %>
  <%= link_to_remove_association raw("<img src='../../../assets/remove.png' width='25px' height='25px'>"), f %>
</div>

我想知道为什么新添加的字段超出了我的edit.html.erb

中的现有字段

2 个答案:

答案 0 :(得分:3)

我认为它会查找带有关联的div作为包装嵌套字段的id。

所以你的观点

<div id="departments">
  <%= f.simple_fields_for :departments do |department| %>
    <%= render 'department_fields', :f => department %>
  <% end %>
  <div class="links">
    <%=link_to_add_association "Add Department", f, :departments,class: "btn btn-inverse" %>
  </div>
</div>

答案 1 :(得分:1)

@pullen的解决方案可行,但他的推理不正确。如果你通过coccon的代码追踪,你会遇到类似这样的东西

var getInsertionNodeElem = function(insertionNode, insertionTraversal, $this){
    if (!insertionNode){
      return $this.parent();
    }
    ...
}

然后我相信这个var addedContent = insertionNodeElem[insertionMethod](contentNode);会在新节点中添加。

重点是新的部分将插入link_to_add_association的父节点之上。因此link_to_add_association的{​​{1}}元素和新输入字段的容器节点将是兄弟姐妹。

如果您希望插入的部分位于下方,则必须将<a>包装在其自己的div中。例如:

link_to_add_association