# encoding: utf-8 require 'action_controller/base' module Cells module Cell # == Basic overview # # A Cell is the central notion of the cells plugin. A cell acts as a # lightweight controller in the sense that it will assign variables and # render a view. Cells can be rendered from other cells as well as from # regular controllers and views (see ActionView::Base#render_cell and # ControllerMethods#render_cell) # # == A render_cell() cycle # # A typical render_cell state rendering cycle looks like this: # render_cell :blog, :newest_article, {...} # - an instance of the class BlogCell is created, and a hash containing # arbitrary parameters is passed # - the state method newest_article is executed and assigns instance # variables to be used in the view # - Usually the state method will call #render and return # - #render will retrieve the corresponding view # (e.g. app/cells/blog/newest_article.html. [erb|haml|...]), # renders this template and returns the markup. # # == Design Principles # A cell is a completely autonomous object and it should not know or have to know # from what controller it is being rendered. For this reason, the controller's # instance variables and params hash are not directly available from the cell or # its views. This is not a bug, this is a feature! It means cells are truly # reusable components which can be plugged in at any point in your application # without having to think about what information is available at that point. # When rendering a cell, you can explicitly pass variables to the cell in the # extra opts argument hash, just like you would pass locals in partials. # This hash is then available inside the cell as the @opts instance variable. # # == Directory hierarchy # # To get started creating your own cells, you can simply create a new directory # structure under your app directory called cells. Cells are # ruby classes which end in the name Cell. So for example, if you have a # cell which manages all user information, it would be called UserCell. # A cell which manages a shopping cart could be called ShoppingCartCell. # # The directory structure of this example would look like this: # app/ # models/ # .. # views/ # .. # helpers/ # application_helper.rb # product_helper.rb # .. # controllers/ # .. # cells/ # shopping_cart_cell.rb # shopping_cart/ # status.html.erb # product_list.html.erb # empty_prompt.html.erb # user_cell.rb # user/ # login.html.erb # layouts/ # box.html.erb # .. # # The directory with the same name as the cell contains views for the # cell's states. A state is an executed method along with a # rendered view, resulting in content. This means that states are to # cells as actions are to controllers, so each state has its own view. # The use of partials is deprecated with cells, it is better to just # render a different state on the same cell (which also works recursively). # # Anyway, render :partial in a cell view will work, if the # partial is contained in the cell's view directory. # # As can be seen above, Cells also can make use of helpers. All Cells # include ApplicationHelper by default, but you can add additional helpers # as well with the ::Cell::Base.helper class method: # class ShoppingCartCell < ::Cell::Base # helper :product # ... # end # # This will make the ProductHelper from app/helpers/product_helper.rb # available from all state views from our ShoppingCartCell. # # == Cell inheritance # # Unlike controllers, Cells can form a class hierarchy. When a cell class # is inherited by another cell class, its states are inherited as regular # methods are, but also its views are inherited. Whenever a view is looked up, # the view finder first looks for a file in the directory belonging to the # current cell class, but if this is not found in the application or any # engine, the superclass' directory is checked. This continues all the # way up until it stops at ::Cell::Base. # # For instance, when you have two cells: # class MenuCell < ::Cell::Base # def show # end # # def edit # end # end # # class MainMenuCell < MenuCell # .. # no need to redefine show/edit if they do the same! # end # and the following directory structure in app/cells: # app/cells/ # menu/ # show.html.erb # edit.html.erb # main_menu/ # show.html.erb # then when you call # render_cell :main_menu, :show # the main menu specific show.html.erb (app/cells/main_menu/show.html.erb) # is rendered, but when you call # render_cell :main_menu, :edit # cells notices that the main menu does not have a specific view for the # edit state, so it will render the view for the parent class, # app/cells/menu/edit.html.erb # # # == Gettext support # # Cells support gettext, just name your views accordingly. It works exactly equivalent # to controller views. # # cells/user/user_form.html.erb # cells/user/user_form_de.html.erb # # If gettext is set to DE_de, the latter view will be chosen. class Base include ::ActionController::Helpers include ::ActionController::RequestForgeryProtection include ActiveHelper class_inheritable_array :view_paths, :instance_writer => false write_inheritable_attribute(:view_paths, ActionView::PathSet.new) # Force use of a PathSet in this attribute, self.view_paths = ActionView::PathSet.new would still yield in an array class << self attr_accessor :request_forgery_protection_token # Use this if you want Cells to look up view templates # in directories other than the default. def view_paths=(paths) self.view_paths.clear.concat(paths) # don't let 'em overwrite the PathSet. end # A template file will be looked for in each view path. This is typically # just RAILS_ROOT/app/cells, but you might want to add e.g. # RAILS_ROOT/app/views. def add_view_path(path) path = File.join(::Rails.root, path) if defined?(::Rails) and ::Rails.respond_to?(:root) self.view_paths << path unless self.view_paths.include?(path) end # Creates a cell instance of the class nameCell, passing through # opts. def create_cell_for(controller, name, opts={}) class_from_cell_name(name).new(controller, opts) end # Declare a controller method as a helper. For example, # helper_method :link_to # def link_to(name, options) ... end # makes the link_to controller method available in the view. def helper_method(*methods) methods.flatten.each do |method| master_helper_module.module_eval <<-end_eval def #{method}(*args, &block) @cell.send(:#{method}, *args, &block) end end_eval end end # Return the default view for the given state on this cell subclass. # This is a file with the name of the state under a directory with the # name of the cell followed by a template extension. def view_for_state(state) "#{cell_name}/#{state}" end # Find a possible template for a cell's current state. It tries to find a # template file with the name of the state under a subdirectory # with the name of the cell under the app/cells directory. # If this file cannot be found, it will try to call this method on # the superclass. This way you only have to write a state template # once when a more specific cell does not need to change anything in # that view. def find_class_view_for_state(state) return [view_for_state(state)] if superclass == ::Cell::Base superclass.find_class_view_for_state(state) << view_for_state(state) end # Get the name of this cell's class as an underscored string, # with _cell removed. # # Example: # UserCell.cell_name # => "user" def cell_name name.underscore.sub(/_cell/, '') end # Given a cell name, finds the class that belongs to it. # # Example: # ::Cell::Base.class_from_cell_name(:user) # => UserCell def class_from_cell_name(cell_name) "#{cell_name}_cell".classify.constantize end def state2view_cache @state2view_cache ||= {} end def cache_configured? ::ActionController::Base.cache_configured? end end class MissingTemplate < ActionView::ActionViewError def initialize(message, possible_paths) super(message + " and possible paths #{possible_paths}") end end class_inheritable_accessor :allow_forgery_protection self.allow_forgery_protection = true class_inheritable_accessor :default_template_format self.default_template_format = :html delegate :params, :session, :request, :logger, :to => :controller attr_accessor :controller attr_reader :state_name def initialize(controller, options={}) @controller = controller @opts = options end def cell_name self.class.cell_name end # Render the given state. You can pass the name as either a symbol or # a string. def render_state(state) @cell = self @state_name = state content = dispatch_state(state) return content if content.kind_of? String render_view_for_backward_compat(content, state) end # Call the state method. def dispatch_state(state) send(state) end # We will soon remove the implicit call to render_view_for, but here it is for your convenience. def render_view_for_backward_compat(opts, state) ::ActiveSupport::Deprecation.warn "You either didn't call #render or forgot to return a string in the state method '#{state}'. However, returning nil is deprecated for the sake of explicitness" render_view_for(opts, state) end # Renders the view for the current state and returns the markup for the component. # Usually called and returned at the end of a state method. # # ==== Options # * :view - Specifies the name of the view file to render. Defaults to the current state name. # * :template_format - Allows using a format different to :html. # * :layout - If set to a valid filename inside your cell's view_paths, the current state view will be rendered inside the layout (as known from controller actions). Layouts should reside in app/cells/layouts. # * :locals - Makes the named parameters available as variables in the view. # * :text - Just renders plain text. # * :inline - Renders an inline template as state view. See ActionView::Base#render for details. # * :file - Specifies the name of the file template to render. # * :nothing - Will make the component kinda invisible and doesn't invoke the rendering cycle. # * :state - Instantly invokes another rendering cycle for the passed state and returns. # Example: # class MyCell < ::Cell::Base # def my_first_state # # ... do something # render # end # # will just render the view my_first_state.html. # # def my_first_state # # ... do something # render :view => :my_first_state, :layout => 'metal' # end # # will also use the view my_first_state.html as template and even put it in the layout # metal that's located at $RAILS_ROOT/app/cells/layouts/metal.html.erb. # # def say_your_name # render :locals => {:name => "Nick"} # end # # will make the variable +name+ available in the view say_your_name.html. # # def say_your_name # render :nothing => true # end # # will render an empty string thus keeping your name a secret. # # # ==== Where have all the partials gone? # In Cells we abandoned the term 'partial' in favor of plain 'views' - we don't need to distinguish # between both terms. A cell view is both, a view and a kind of partial as it represents only a small # part of the page. # Just use :view and enjoy. def render(opts={}) render_view_for(opts, @state_name) ### FIXME: i don't like the magic access to @state_name here. ugly! end # Render the view belonging to the given state. Will raise ActionView::MissingTemplate # if it can not find one of the requested view template. Note that this behaviour was # introduced in cells 2.3 and replaces the former warning message. def render_view_for(opts, state) return '' if opts[:nothing] action_view = setup_action_view ### TODO: dispatch dynamically: if opts[:text] elsif opts[:inline] elsif opts[:file] elsif opts[:state] opts[:text] = render_state(opts[:state]) else # handle :layout, :template_format, :view opts = defaultize_render_options_for(opts, state) # set instance vars, include helpers: prepare_action_view_for(action_view, opts) template = find_family_view_for_state_with_caching(opts[:view], action_view) opts[:file] = template end opts = sanitize_render_options(opts) action_view.render_for(opts) end # Defaultize the passed options from #render. def defaultize_render_options_for(opts, state) opts[:template_format] ||= self.class.default_template_format opts[:view] ||= state opts end def prepare_action_view_for(action_view, opts) # make helpers available: include_helpers_in_class(action_view.class) import_active_helpers_into(action_view) # in Cells::Cell::ActiveHelper. action_view.assigns = assigns_for_view # make instance vars available. action_view.template_format = opts[:template_format] end def setup_action_view view_class = Class.new(::Cells::Cell::View) action_view = view_class.new(self.class.view_paths, {}, @controller) action_view.cell = self action_view end # Prepares opts to be passed to ActionView::Base#render by removing # unknown parameters. def sanitize_render_options(opts) opts.except!(:view, :state) end # Climbs up the inheritance hierarchy of the Cell, looking for a view # for the current state in each level. # As soon as a view file is found it is returned as an ActionView::Template # instance. ### DISCUSS: moved to Cell::View#find_template in rainhead's fork: def find_family_view_for_state(state, action_view) missing_template_exception = nil possible_paths = possible_paths_for_state(state) possible_paths.each do |template_path| # we need to catch MissingTemplate, since we want to try for all possible # family views. begin if view = action_view.try_picking_template_for_path(template_path) return view end rescue ::ActionView::MissingTemplate => missing_template_exception end end raise MissingTemplate.new(missing_template_exception.message, possible_paths) end # In production mode, the view for a state/template_format is cached. ### DISCUSS: ActionView::Base already caches results for #pick_template, so maybe ### we should just cache the family path for a state/format? def find_family_view_for_state_with_caching(state, action_view) return find_family_view_for_state(state, action_view) unless self.class.cache_configured? # in production mode: key = "#{state}/#{action_view.template_format}" state2view = self.class.state2view_cache state2view[key] || state2view[key] = find_family_view_for_state(state, action_view) end # Find possible files that belong to the state. This first tries the cell's # #view_for_state method and if that returns a true value, it # will accept that value as a string and interpret it as a pathname for # the view file. If it returns a falsy value, it will call the Cell's class # method find_class_view_for_state to determine the file to check. # # You can override the ::Cell::Base#view_for_state method for a particular # cell if you wish to make it decide dynamically what file to render. def possible_paths_for_state(state) self.class.find_class_view_for_state(state).reverse! end # Prepares the hash {instance_var => value, ...} that should be available # in the ActionView when rendering the state view. def assigns_for_view assigns = {} (self.instance_variables - ivars_to_ignore).each do |k| assigns[k[1..-1]] = instance_variable_get(k) end assigns end # When passed a copy of the ActionView::Base class, it # will mix in all helper classes for this cell in that class. def include_helpers_in_class(view_klass) view_klass.send(:include, self.class.master_helper_module) end # Defines the instance variables that should not be copied to the # View instance. def ivars_to_ignore; ['@controller']; end ### TODO: allow log levels. def log(message) return unless @controller.logger @controller.logger.debug(message) end end end end