require 'ostruct' # PageTitleHelper module PageTitleHelper module Interpolations extend self # Returns a sorted list of all interpolations. def self.all self.instance_methods(false).sort end def self.interpolate(pattern, *args) all.reverse.inject(pattern.dup) do |result, tag| result.gsub(/:#{tag}/) do |match| send(tag, *args) end end end def app(env) env.options[:app] || I18n.translate(:'app.name', :default => File.basename(RAILS_ROOT).humanize) end def title(env) env.title end end # Add new, custom, interpolation. def self.interpolates(key, &block) Interpolations.send(:define_method, key, &block) end # Default options, which are globally referenced and can # be changed globally, which might be useful in some cases. def self.options @options ||= { :format => ':app - :title', :default => :'app.tagline', :suffix => :title } end def page_title(options = nil, &block) if block_given? content_for(:page_title) { yield } return read_page_title_content_block end options = PageTitleHelper.options.merge(options || {}) options.assert_valid_keys(:app, :suffix, :default, :format) # construct basic env env = OpenStruct.new(:options => options, :view => self, :controller => self.controller) # just return the applications name return Interpolations.app(env) if options[:format] == :app # read page title page_title = read_page_title_content_block page_title = I18n.translate(i18n_template_key(options[:suffix]), :default => options[:default]) if page_title.blank? # return page title if format is set explicitly set to false return page_title if options[:format] == false # else -> interpolate env.title = page_title Interpolations.interpolate options[:format], env end protected # Access @content_for_page_title variable, though this is a tad # hacky, because... what if they (the rails guys) change the behaviour of # content_for? Well, in Rails 2.3.x it works so far. # # But to simplify compatibility with later versions, this method kinda abstracts # away access to the content within a content_for block. def read_page_title_content_block instance_variable_get(:'@content_for_page_title') end # Access +ActionView+s internal @_first_render variable, to access # template first rendered, this is to help create the DRY-I18n-titles magic, # and also kind of a hack, because this really seems to be some sort if # internal variable, that's why it's "abstracted" away as well. # def read_first_render_path @_first_render.template_path end def i18n_template_key(suffix = nil) ikey = read_first_render_path.gsub(/\.html\.erb$/, '').tr('/', '.') ikey = ikey + "." + suffix.to_s unless suffix.nil? ikey end end # tie stuff together if Object.const_defined?('ActionView') ActionView::Base.send(:include, PageTitleHelper) end