aboutsummaryrefslogtreecommitdiffstats
path: root/actionpack/lib/action_view/lookup_context.rb
diff options
context:
space:
mode:
authorJosé Valim <jose.valim@gmail.com>2010-03-08 20:39:15 +0100
committerJosé Valim <jose.valim@gmail.com>2010-03-08 20:39:15 +0100
commitbdf5096816d03f2bdaefd20a07a0fa562543549c (patch)
tree3e656b06836772e74f8c44492d61caddb6bfb893 /actionpack/lib/action_view/lookup_context.rb
parent67a6725bf934452219cc054c8cd0535148d4fcd4 (diff)
downloadrails-bdf5096816d03f2bdaefd20a07a0fa562543549c.tar.gz
rails-bdf5096816d03f2bdaefd20a07a0fa562543549c.tar.bz2
rails-bdf5096816d03f2bdaefd20a07a0fa562543549c.zip
Move details to lookup_context and make resolvers use the cache key.
Diffstat (limited to 'actionpack/lib/action_view/lookup_context.rb')
-rw-r--r--actionpack/lib/action_view/lookup_context.rb157
1 files changed, 103 insertions, 54 deletions
diff --git a/actionpack/lib/action_view/lookup_context.rb b/actionpack/lib/action_view/lookup_context.rb
index e259a78c5c..231882185f 100644
--- a/actionpack/lib/action_view/lookup_context.rb
+++ b/actionpack/lib/action_view/lookup_context.rb
@@ -1,14 +1,30 @@
+require 'active_support/core_ext/object/try'
+
module ActionView
# LookupContext is the object responsible to hold all information required to lookup
# templates, i.e. view paths and details. The LookupContext is also responsible to
# generate a key, given to view paths, used in the resolver cache lookup. Since
# this key is generated just once during the request, it speeds up all cache accesses.
class LookupContext #:nodoc:
- attr_reader :details, :view_paths
-
mattr_accessor :fallbacks
@@fallbacks = [FileSystemResolver.new(""), FileSystemResolver.new("/")]
+ mattr_accessor :registered_details
+ self.registered_details = {}
+
+ def self.register_detail(name, options = {})
+ registered_details[name] = lambda do |value|
+ value = (value.blank? || options[:accessible] == false) ?
+ Array(yield) : Array(value)
+ value |= [nil] unless options[:allow_nil] == false
+ value
+ end
+ end
+
+ register_detail(:formats) { Mime::SET.symbols }
+ register_detail(:locale, :accessible => false) { [I18n.locale] }
+ register_detail(:handlers, :accessible => false) { Template::Handlers.extensions }
+
class DetailsKey #:nodoc:
attr_reader :details
alias :eql? :equal?
@@ -22,75 +38,108 @@ module ActionView
def initialize(details)
@details, @hash = details, details.hash
end
+
+ def outdated?(details)
+ @details != details
+ end
end
def initialize(view_paths, details = {})
- @details, @details_key = details, nil
self.view_paths = view_paths
+ self.details = details
+ @details_key = nil
end
- # Shortcut to read formats from details.
- def formats
- @details[:formats]
- end
-
- # Shortcut to set formats in details.
- def formats=(value)
- self.details = @details.merge(:formats => Array(value))
- end
+ module ViewPaths
+ attr_reader :view_paths
- # Whenever setting view paths, makes a copy so we can manipulate then in
- # instance objects as we wish.
- def view_paths=(paths)
- @view_paths = ActionView::Base.process_view_paths(paths)
- end
+ # Whenever setting view paths, makes a copy so we can manipulate then in
+ # instance objects as we wish.
+ def view_paths=(paths)
+ @view_paths = ActionView::Base.process_view_paths(paths)
+ end
- # Setter for details. Everything this method is invoked, we need to nullify
- # the details key if it changed.
- def details=(details)
- @details = details
- @details_key = nil if @details_key && @details_key.details != details
- end
+ def find_template(name, prefix = nil, partial = false)
+ key = details_key
+ @view_paths.find(name, key.details, prefix, partial || false, key)
+ end
- def details_key
- @details_key ||= DetailsKey.get(details) unless details.empty?
- end
+ def template_exists?(name, prefix = nil, partial = false)
+ key = details_key
+ @view_paths.exists?(name, key.details, prefix, partial || false, key)
+ end
- # Update the details keys by merging the given hash into the current
- # details hash. If a block is given, the details are modified just during
- # the execution of the block and reverted to the previous value after.
- def update_details(new_details)
- old_details = self.details
- self.details = old_details.merge(new_details)
-
- if block_given?
- begin
- yield
- ensure
- self.details = old_details
+ # Add fallbacks to the view paths. Useful in cases you are rendering a file.
+ def with_fallbacks
+ added_resolvers = 0
+ self.class.fallbacks.each do |resolver|
+ next if view_paths.include?(resolver)
+ view_paths.push(resolver)
+ added_resolvers += 1
end
+ yield
+ ensure
+ added_resolvers.times { view_paths.pop }
end
end
- # Added fallbacks to the view paths. Useful in cases you are rendering a file.
- def with_fallbacks
- added_resolvers = 0
- self.class.fallbacks.each do |resolver|
- next if view_paths.include?(resolver)
- view_paths.push(resolver)
- added_resolvers += 1
+ module Details
+ def details
+ @details = normalize_details(@details)
end
- yield
- ensure
- added_resolvers.times { view_paths.pop }
- end
- def find_template(name, prefix = nil, partial = false)
- @view_paths.find(name, details, prefix, partial || false, details_key)
- end
+ def details=(new_details)
+ @details = new_details
+ details
+ end
- def template_exists?(name, prefix = nil, partial = false)
- @view_paths.exists?(name, details, prefix, partial || false, details_key)
+ # TODO This is too expensive. Revisit this.
+ def details_key
+ latest_details = self.details
+ @details_key = nil if @details_key.try(:outdated?, latest_details)
+ @details_key ||= DetailsKey.get(latest_details)
+ end
+
+ # Shortcut to read formats from details.
+ def formats
+ self.details[:formats]
+ end
+
+ # Shortcut to set formats in details.
+ def formats=(value)
+ self.details = @details.merge(:formats => value)
+ end
+
+ # Update the details keys by merging the given hash into the current
+ # details hash. If a block is given, the details are modified just during
+ # the execution of the block and reverted to the previous value after.
+ def update_details(new_details)
+ old_details = self.details
+ self.details = old_details.merge(new_details)
+
+ if block_given?
+ begin
+ yield
+ ensure
+ self.details = old_details
+ end
+ end
+ end
+
+ protected
+
+ def normalize_details(details)
+ details = details.dup
+ # TODO: Refactor this concern out of the resolver
+ details.delete(:formats) if details[:formats] == [:"*/*"]
+ self.class.registered_details.each do |k, v|
+ details[k] = v.call(details[k])
+ end
+ details
+ end
end
+
+ include Details
+ include ViewPaths
end
end \ No newline at end of file