2010-03-08 08:46:57 -05:00
|
|
|
module ActionView
|
|
|
|
module Layouts
|
|
|
|
# You can think of a layout as a method that is called with a block. _layout_for
|
|
|
|
# returns the contents that are yielded to the layout. If the user calls yield
|
|
|
|
# :some_name, the block, by default, returns content_for(:some_name). If the user
|
|
|
|
# calls yield, the default block returns content_for(:layout).
|
|
|
|
#
|
|
|
|
# The user can override this default by passing a block to the layout.
|
|
|
|
#
|
|
|
|
# ==== Example
|
|
|
|
#
|
|
|
|
# # The template
|
2010-03-12 07:39:47 -05:00
|
|
|
# <%= render :layout => "my_layout" do %>Content<% end %>
|
2010-03-08 08:46:57 -05:00
|
|
|
#
|
|
|
|
# # The layout
|
|
|
|
# <html><% yield %></html>
|
|
|
|
#
|
|
|
|
# In this case, instead of the default block, which would return content_for(:layout),
|
|
|
|
# this method returns the block that was passed in to render layout, and the response
|
|
|
|
# would be <html>Content</html>.
|
|
|
|
#
|
|
|
|
# Finally, the block can take block arguments, which can be passed in by yield.
|
|
|
|
#
|
|
|
|
# ==== Example
|
|
|
|
#
|
|
|
|
# # The template
|
2010-03-12 07:39:47 -05:00
|
|
|
# <%= render :layout => "my_layout" do |customer| %>Hello <%= customer.name %><% end %>
|
2010-03-08 08:46:57 -05:00
|
|
|
#
|
|
|
|
# # The layout
|
|
|
|
# <html><% yield Struct.new(:name).new("David") %></html>
|
|
|
|
#
|
|
|
|
# In this case, the layout would receive the block passed into <tt>render :layout</tt>,
|
|
|
|
# and the Struct specified in the layout would be passed into the block. The result
|
|
|
|
# would be <html>Hello David</html>.
|
|
|
|
def _layout_for(name = nil, &block) #:nodoc:
|
|
|
|
if !block || name
|
|
|
|
@_content_for[name || :layout]
|
|
|
|
else
|
|
|
|
capture(&block)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# This is the method which actually finds the layout using details in the lookup
|
|
|
|
# context object. If no layout is found, it checkes if at least a layout with
|
|
|
|
# the given name exists across all details before raising the error.
|
2010-03-16 18:35:45 -04:00
|
|
|
#
|
|
|
|
# If self.formats contains several formats, just the first one is considered in
|
|
|
|
# the layout lookup.
|
|
|
|
def find_layout(layout)
|
2010-03-08 08:46:57 -05:00
|
|
|
begin
|
2010-03-16 18:35:45 -04:00
|
|
|
if formats.size == 1
|
|
|
|
_find_layout(layout)
|
|
|
|
else
|
2010-03-19 12:20:15 -04:00
|
|
|
update_details(:formats => self.formats.first){ _find_layout(layout) }
|
2010-03-16 18:35:45 -04:00
|
|
|
end
|
2010-03-08 08:46:57 -05:00
|
|
|
rescue ActionView::MissingTemplate => e
|
|
|
|
update_details(:formats => nil) do
|
2010-03-12 08:25:10 -05:00
|
|
|
raise unless template_exists?(layout)
|
2010-03-08 08:46:57 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2010-03-16 18:35:45 -04:00
|
|
|
def _find_layout(layout) #:nodoc:
|
|
|
|
layout =~ /^\// ?
|
|
|
|
with_fallbacks { find_template(layout) } : find_template(layout)
|
|
|
|
end
|
|
|
|
|
2010-03-08 08:46:57 -05:00
|
|
|
# Contains the logic that actually renders the layout.
|
|
|
|
def _render_layout(layout, locals, &block) #:nodoc:
|
|
|
|
layout.render(self, locals){ |*name| _layout_for(*name, &block) }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|