aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--actionpack/CHANGELOG2
-rw-r--r--actionpack/lib/action_view/base.rb89
2 files changed, 50 insertions, 41 deletions
diff --git a/actionpack/CHANGELOG b/actionpack/CHANGELOG
index 860c35b768..a9ff040070 100644
--- a/actionpack/CHANGELOG
+++ b/actionpack/CHANGELOG
@@ -4,7 +4,7 @@
* Added support for per-action session management #1763
-* Improved rendering speed on complicated templates by up to 25% #1234 [Stephan Kaes]. This did necessasitate a change to the internals of ActionView#render_template that now has four parameters. Developers of custom view handlers (like Amrita) need to update for that.
+* Improved rendering speed on complicated templates by up to 100% (the more complex the templates, the higher the speedup) #1234 [Stephan Kaes]. This did necessasitate a change to the internals of ActionView#render_template that now has four parameters. Developers of custom view handlers (like Amrita) need to update for that.
* Added options hash as third argument to FormHelper#input, so you can do input('person', 'zip', :size=>10) #1719 [jeremye@bsa.ca.gov]
diff --git a/actionpack/lib/action_view/base.rb b/actionpack/lib/action_view/base.rb
index f20620d4f0..9f21f3fe3b 100644
--- a/actionpack/lib/action_view/base.rb
+++ b/actionpack/lib/action_view/base.rb
@@ -118,15 +118,17 @@ module ActionView #:nodoc:
# More builder documentation can be found at http://builder.rubyforge.org.
class Base
include ERB::Util
-
+
attr_reader :first_render
attr_accessor :base_path, :assigns, :template_extension
attr_accessor :controller
-
+
attr_reader :logger, :params, :response, :session, :headers, :flash
- # Turn on to cache the reading of templates from the file system. Doing so means that you have to restart the server
- # when changing templates, but that rendering will be faster.
+ # Turn on to cache the reading of templates from the file system.
+ # Doing so means that you have to restart the server when changing
+ # templates, but it will save checking whether the file has changed
+ # on disk.
@@cache_template_loading = false
cattr_accessor :cache_template_loading
@@ -135,8 +137,8 @@ module ActionView #:nodoc:
@@erb_trim_mode = '-'
cattr_accessor :erb_trim_mode
- @@compiled_erb_templates = {}
- @@erb_count = 0
+ @@compiled_templates = {}
+ @@template_count = 0
@@loaded_templates = {}
@@template_handlers = {}
@@ -165,7 +167,7 @@ module ActionView #:nodoc:
# is made available as local variables.
def render_file(template_path, use_full_path = true, local_assigns = {})
@first_render = template_path if @first_render.nil?
-
+
if use_full_path
template_extension = pick_template_extension(template_path)
template_file_name = full_template_path(template_path, template_extension)
@@ -173,7 +175,7 @@ module ActionView #:nodoc:
template_file_name = template_path
template_extension = template_path.split('.').last
end
-
+
template_source = read_template_file(template_file_name, template_extension)
begin
@@ -187,7 +189,7 @@ module ActionView #:nodoc:
end
end
end
-
+
# Renders the template present at <tt>template_path</tt> (relative to the template_root).
# The hash in <tt>local_assigns</tt> is made available as local variables.
def render(options = {}, old_local_assigns = {})
@@ -196,7 +198,7 @@ module ActionView #:nodoc:
elsif options.is_a?(Hash)
options[:locals] ||= {}
options[:use_full_path] = options[:use_full_path].nil? ? true : options[:use_full_path]
-
+
if options[:file]
render_file(options[:file], options[:use_full_path], options[:locals])
elsif options[:partial] && options[:collection]
@@ -208,7 +210,7 @@ module ActionView #:nodoc:
end
end
end
-
+
# Renders the +template+ which is given as a string as either rhtml or rxml depending on <tt>template_extension</tt>.
# The hash in <tt>local_assigns</tt> is made available as local variables.
def render_template(template_extension, template, file_name = nil, local_assigns = {})
@@ -232,7 +234,7 @@ module ActionView #:nodoc:
raise ActionViewError, "No rhtml, rxml, or delegate template found for #{template_path}"
end
end
-
+
def delegate_template_exists?(template_path)#:nodoc:
@@template_handlers.find { |k,| template_exists?(template_path, k) }
end
@@ -271,70 +273,77 @@ module ActionView #:nodoc:
!@@cache_template_loading )
if read_file
@@loaded_templates[template_path] = info = File.read(template_path)
- @@compiled_erb_templates[template_path] = nil if 'rhtml' == extension
+ @@compiled_templates[template_path] = nil
end
info
end
def evaluate_assigns(local_assigns = {})
- @assigns.each { |key, value| instance_variable_set "@#{key}", value }
+ @assigns.each { |key, value| instance_variable_set("@#{key}", value) }
saved_locals = {}
local_assigns.each do |key, value|
varstr = "@_#{key}_"
- saved_locals[varstr] = instance_variable_get varstr
- instance_variable_set varstr, value
- unless self.respond_to? key
- self.class.class_eval "def #{key}; #{varstr}; end"
- self.class.class_eval "def #{key}=(v); #{varstr} = v; end"
+ saved_locals[varstr] = instance_variable_get(varstr)
+ instance_variable_set(varstr, value)
+ unless self.respond_to?(key)
+ self.class.class_eval("def #{key}; #{varstr}; end")
+ self.class.class_eval("def #{key}=(v); #{varstr} = v; end")
end
end
saved_locals
end
- def compile_erb_template(template, file_name)
+ def compile_template(extension, template, file_name)
cache_name = file_name || template
-
- unless @@compiled_erb_templates[cache_name]
- erb = ERB.new(template, nil, @@erb_trim_mode)
- erb_name = 'run_erb_'
+ unless @@compiled_templates[cache_name]
+ t_name, t_arg, t_code = nil
+ case extension
+ when :rhtml
+ t_name = 'run_html_'
+ t_arg = ''
+ t_code = ERB.new(template, nil, @@erb_trim_mode).src
+ when :rxml
+ t_name = 'run_xml_'
+ t_arg = '(xml)'
+ t_code = template
+ end
if file_name
i = file_name.index(@base_path)
l = @base_path.length
s_file_name = i ? file_name[i+l+1,file_name.length-l-1] : file_name
- s_file_name.sub!(/.rhtml$/,'')
+ s_file_name.sub!(/(.rhtml|.rxml)$/,'')
s_file_name.tr!('/:-', '_')
s_file_name.gsub!(/[^a-zA-Z0-9_]/){|s| s[0].to_s}
- erb_name += s_file_name
+ t_name += s_file_name
else
- @@erb_count += 1
- erb_name += @@erb_count.to_s
+ @@template_count += 1
+ t_name += @@template_count.to_s
end
- erb_def = "def #{erb_name}; #{erb.src}; end"
- eval erb_def rescue raise ActionViewError, "ERROR defining #{erb_name}: #{erb_def}"
+ t_def = "def #{t_name}#{t_arg}; #{t_code}; end"
+ self.class.class_eval(t_def) rescue raise ActionViewError, "ERROR defining #{t_name}: #{t_def}"
- @@compiled_erb_templates[cache_name] = erb_name.intern
+ @@compiled_templates[cache_name] = t_name.intern
@@loaded_templates[cache_name] = Time.now if file_name
- logger.debug "Compiled erb template #{cache_name}\n ==> #{erb_name}" if logger
+ logger.info "Compiled template #{cache_name}\n ==> #{t_name}" if logger
end
-
- @@compiled_erb_templates[cache_name]
+ @@compiled_templates[cache_name]
end
def rhtml_render(extension, template, file_name, local_assigns)
- render_sym = compile_erb_template(template, file_name)
+ render_sym = compile_template(:rhtml, template, file_name)
saved_locals = evaluate_assigns(local_assigns)
result = self.send(render_sym)
- saved_locals.each { |k,v| instance_variable_set(k, v) }
+ saved_locals.each{ |k,v| instance_variable_set(k, v) }
result
end
def rxml_render(extension, template, file_name, local_assigns)
@controller.headers["Content-Type"] ||= 'text/xml'
+ render_sym = compile_template(:rxml, template, file_name)
saved_locals = evaluate_assigns(local_assigns)
- xml = Builder::XmlMarkup.new(:indent => 2)
- result = eval(template, binding, '(template)(eval)', 1)
- saved_locals.each { |k,v| instance_variable_set(k,v) }
+ result = self.send(render_sym, Builder::XmlMarkup.new(:indent => 2))
+ saved_locals.each{ |k,v| instance_variable_set(k, v) }
result
end
@@ -344,4 +353,4 @@ module ActionView #:nodoc:
end
end
-require 'action_view/template_error'
+require 'action_view/template_error' \ No newline at end of file