aboutsummaryrefslogtreecommitdiffstats
path: root/actionpack/lib/action_view/template.rb
diff options
context:
space:
mode:
Diffstat (limited to 'actionpack/lib/action_view/template.rb')
-rw-r--r--actionpack/lib/action_view/template.rb138
1 files changed, 84 insertions, 54 deletions
diff --git a/actionpack/lib/action_view/template.rb b/actionpack/lib/action_view/template.rb
index 5e5ea9b9b9..03f9234289 100644
--- a/actionpack/lib/action_view/template.rb
+++ b/actionpack/lib/action_view/template.rb
@@ -1,82 +1,112 @@
module ActionView #:nodoc:
- class Template #:nodoc:
+ class Template
extend TemplateHandlers
include Renderable
- attr_reader :path, :extension
+ attr_accessor :filename, :load_path, :base_path, :name, :format, :extension
+ delegate :to_s, :to => :path
- def initialize(view, path, use_full_path = nil, locals = {})
- unless use_full_path == nil
- ActiveSupport::Deprecation.warn("use_full_path option has been deprecated and has no affect.", caller)
- end
+ def initialize(template_path, load_paths = [])
+ template_path = template_path.dup
+ @base_path, @name, @format, @extension = split(template_path)
+ @base_path.to_s.gsub!(/\/$/, '') # Push to split method
+ @load_path, @filename = find_full_path(template_path, load_paths)
+
+ # Extend with partial super powers
+ extend RenderablePartial if @name =~ /^_/
+ end
+
+ def freeze
+ # Eager load memoized methods
+ format_and_extension
+ path
+ path_without_extension
+ path_without_format_and_extension
+ source
+ method_segment
+
+ # Eager load memoized methods from Renderable
+ handler
+ compiled_source
+
+ instance_variables.each { |ivar| ivar.freeze }
+
+ super
+ end
- @view = view
- @paths = view.view_paths
+ def format_and_extension
+ @format_and_extension ||= (extensions = [format, extension].compact.join(".")).blank? ? nil : extensions
+ end
+
+ def path
+ @path ||= [base_path, [name, format, extension].compact.join('.')].compact.join('/')
+ end
- @original_path = path
- @path = TemplateFile.from_path(path)
- @view.first_render ||= @path.to_s
+ def path_without_extension
+ @path_without_extension ||= [base_path, [name, format].compact.join('.')].compact.join('/')
+ end
- set_extension_and_file_name
+ def path_without_format_and_extension
+ @path_without_format_and_extension ||= [base_path, name].compact.join('/')
+ end
- @method_segment = compiled_method_name_file_path_segment
- @locals = (locals && locals.dup) || {}
- @handler = self.class.handler_class_for_extension(@extension).new(@view)
+ def source
+ @source ||= File.read(@filename)
end
- def render_template
- render
+ def method_segment
+ unless @method_segment
+ segment = File.expand_path(@filename)
+ segment.sub!(/^#{Regexp.escape(File.expand_path(RAILS_ROOT))}/, '') if defined?(RAILS_ROOT)
+ segment.gsub!(/([^a-zA-Z0-9_])/) { $1.ord }
+ @method_segment = segment
+ end
+
+ @method_segment
+ end
+
+ def render_template(view, local_assigns = {})
+ render(view, local_assigns)
rescue Exception => e
raise e unless filename
if TemplateError === e
e.sub_template_of(filename)
raise e
else
- raise TemplateError.new(self, @view.assigns, e)
+ raise TemplateError.new(self, view.assigns, e)
end
end
- def source
- @source ||= File.read(@filename)
- end
-
- def base_path_for_exception
- (@paths.find_load_path_for_path(@path) || @paths.first).to_s
- end
-
private
- def set_extension_and_file_name
- @extension = @path.extension
-
- unless @extension
- @path = @view.send(:template_file_from_name, @path)
- raise_missing_template_exception unless @path
- @extension = @path.extension
- end
-
- if p = @paths.find_template_file_for_path(path)
- @path = p
- @filename = @path.full_path
- @extension = @path.extension
- raise_missing_template_exception if @filename.blank?
- else
- @filename = @original_path
- raise_missing_template_exception unless File.exist?(@filename)
- end
+ def valid_extension?(extension)
+ Template.template_handler_extensions.include?(extension)
end
- def raise_missing_template_exception
- full_template_path = @original_path.include?('.') ? @original_path : "#{@original_path}.#{@view.template_format}.erb"
- display_paths = @paths.join(':')
- template_type = (@original_path =~ /layouts/i) ? 'layout' : 'template'
- raise MissingTemplate, "Missing #{template_type} #{full_template_path} in view path #{display_paths}"
+ def find_full_path(path, load_paths)
+ load_paths = Array(load_paths) + [nil]
+ load_paths.each do |load_path|
+ file = [load_path, path].compact.join('/')
+ return load_path, file if File.exist?(file)
+ end
+ raise MissingTemplate.new(load_paths, path)
end
- def compiled_method_name_file_path_segment
- s = File.expand_path(@filename)
- s.sub!(/^#{Regexp.escape(File.expand_path(RAILS_ROOT))}/, '') if defined?(RAILS_ROOT)
- s.gsub!(/([^a-zA-Z0-9_])/) { $1.ord }
- s
+ # Returns file split into an array
+ # [base_path, name, format, extension]
+ def split(file)
+ if m = file.match(/^(.*\/)?([^\.]+)\.?(\w+)?\.?(\w+)?\.?(\w+)?$/)
+ if m[5] # Mulipart formats
+ [m[1], m[2], "#{m[3]}.#{m[4]}", m[5]]
+ elsif m[4] # Single format
+ [m[1], m[2], m[3], m[4]]
+ else
+ if valid_extension?(m[3]) # No format
+ [m[1], m[2], nil, m[3]]
+ else # No extension
+ [m[1], m[2], m[3], nil]
+ end
+ end
+ end
end
end
end