在Rails中使用多个yield的部分视图

5
我正在寻找一个解决方案,可以在具有多个 yields 的部分中使用。
在实际例子中,我有这样的视图结构:
基本应用程序.erb(/views/layouts/application.erb):
<!DOCTYPE html>
<head>
    <title>Some title</title>
</head>
<body>
<div class="page">
    <%= yield %>
</div>
</body>
</html> 

一些偏爱 DRY 的代码(/views/shared/content.erb):
<div class="content">
    <div class="sidebar">
        <%= yield :sidebar %>
    </div>
    <div class="main">
        <%= yield %>
    </div>
</div>

控制器视图(/views/home/index.erb):
<%= render :partial => 'layouts/header' %>    
<%= render :partial => 'shared/navigation' %>

<% # It is close to what I want to do %>
<%= render :layout => 'shared/content' do %>
    <% content_for :sidebar do %>
        <%# This is will go to application.erb, not in content.erb %>
        <%= render :partial => 'shared/menu' %>
    <% end %>

    <%= yield %>
<% end %>

<%= render :partial => 'layouts/footer' %>

所以这里的主要问题是有一个带有多个 yield 区域的模板块,并且可以传递自定义的 HTML 或渲染另一个 partial。
3个回答

14

这个问题虽然旧了,但当我在谷歌上搜答案时,仍然很相关。

我想出了一个解决方案,虽然不太优美,但非常有效。这个想法使用Rails的capture方法,它接受一个块并将其内容存储到变量中:

controller.html.erb

<%= render 'shared/partial', body: capture { %>
  My body content
<% }, footer: capture { %>
  My footer content
<% } %>

shared/_partial.html.erb

<div id="body"><%= body %></div>
<div id="footer"><%= footer %></div>

希望这能帮助到某人!


4
在我的情况下,我已经找到了这样的解决方案。
在我的控制器视图(/views/home/index.erb)中:
<% sidebar_content = render :partial => 'shared/slider' %>
<%= render :layout => 'shared/content', :locals => {:sidebar => sidebar_content} do %>
    <%= yield %>
<% end %>

多区域的partial文件(/views/shared/content.erb):

<div class="content">
    <div class="sidebar">
        <%= sidebar %>
    </div>
    <div class="main">
        <%= yield %>
    </div>
</div>

这个解决方案看起来不太美观,但它是可行的。我希望在不久的将来能找到更好的方案。

不用了。我已经改用模板来处理我的工作流程了。我只使用局部模板,并试图避免将另一个局部模板作为参数传递。特别是在上面提到的例子中,我将创建一个带有菜单的“content”局部模板。 /shared/content/default.erb/shared/content/with_menu.erb - a.s.panchenko

1
这是我的解决方案:

/views/shared/_content.erb

<div class="content">
    <div class="sidebar">
        <%= yield :sidebar %>
    </div>
    <div class="main">
        <%= yield %>
    </div>
</div>

views/home/index.erb


<%= my_content_tag do %>
    <% content_for :sidebar do %>
        <%= render :partial => 'shared/menu' %>
    <% end %>

    <%= yield %>
<% end %>

app/helpers/my_tags_helper.rb

module MyTagsHelper
  def my_content_tag(&block)
    sidebar_backup = @view_flow.content.delete(:sidebar)
    x = capture(&block)
    html = render('shared/content') { x }
    @view_flow.content[:sidebar] = sidebar_backup if sidebar_backup
    html
  end
end

关键在于使用capture提取content_for块,并将其传递到@view_flow中。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接