module Lolita module Configuration # Tab is used as a container for different type of resource content. # It can contain fields, images, audio, video files, maps and other stuff. # Tab contain following parts, that also are configurable through block or # arguments passed to tab. # * nested_fields_for Create fields for different DB table/collection than given. # * field_set Create fields set for ease use in builders. # * field Create field in tab, fields also can be included in #nested_fields_for and # #field_set # * default_fields # To create new tab you should pass Lolita::DBI object and tab type, default is :content # ====Example # Lolita::Configuration::Tab.new(Lolita::DBI.new(Post),:images) # To define tab in ORM model, through lolita configuration do the following # ====Example # lolita do # tab(:files) # end module Tab class Base include Lolita::Builder # For different types there are different builders(cells) @@available_types=[:content] lolita_accessor :title,:name,:type, :dissociate attr_accessor :dbi,:current_fieldset, :current_nested_form,:current_dbi attr_reader :field_sets,:nested_forms # To create new tab the following parametrs need to be provided. # * dbi Lolita::DBI::Base object, that represents database. # * *args See #set_attributes, for how these args are processed. # * &block Block can be passed, anything in block will be evaled for current instance. def initialize dbi,type,*args,&block @fields = Lolita::Configuration::Fields.new @field_sets = [] @nested_forms = [] @type = type self.dbi=dbi self.current_dbi=dbi self.set_attributes(*args) self.instance_eval(&block) if block_given? set_default_attributes end # Field setter method, accpet *args and &block to be passed. # For details how to pass args and block see Lolita::Configuration::Field. # Return field itself. def field *args, &block field=Lolita::Configuration::Factory::Field.add(self.current_dbi,*args,&block) field.field_set = current_fieldset if self.current_dbi!=self.dbi field.nested_in=self.dbi field.nested_form = current_nested_form end @fields << field field end # Return all fields in tab. def fields @fields end # Set all fields in tab. Accept fields as Array. # Each array element can be Lolita::Configuration::Field object or # Hash, that will be passed to #field method. def fields=(fields) @fields=Lolita::Configuration::Fields.new if fields.is_a?(Array) fields.each{|field_attr| if field_attr.is_a?(Lolita::Configuration::Field) @fields< db_field.name, :type => db_field.type, :dbi_field => db_field) if db_field.content? } end # Add tab nested fields for class_or_name and &block # that will be evaluted in current tab instance. def nested_fields_for class_or_name, options ={},&block current_class = get_class(class_or_name) nested_form = Lolita::Configuration::NestedForm.new(self,class_or_name, options) self.current_nested_form = nested_form @nested_forms << nested_form self.current_dbi = Lolita::DBI::Base.create(current_class) self.instance_eval(&block) self.current_dbi = self.dbi self.current_nested_form = nil end # Return nested field for given class_or_name def nested_fields_of class_or_name current_class=get_class(class_or_name) self.fields.select{|field| field.nested_in?(@dbi) && field.dbi.klass==current_class } end # Create new field_set for current tab with given _name_ and &block # that will be evaluted in current tab instance. def field_set name,&block field_set=Lolita::Configuration::FieldSet.new(self,name) self.current_fieldset = field_set @field_sets << field_set self.instance_eval(&block) self.current_fieldset=nil end def fields_with_field_set used_fieldsets=[] self.fields.each{|field| if !field.field_set || (!used_fieldsets.include?(field.field_set)) if field.field_set yield field.field_set.fields,field.field_set used_fieldsets<*args. # First element of args is used as type other interpreted as options. # Every Hash key is used as setter method, and value as method value. # ====Example # set_attributes(:content,:field=>{:name=>"My Field"}) # set_attributes(:field=>{:name=>"My Field"}) def set_attributes *args if args options=args.extract_options! options.each{|method,options| self.send(:"#{method}=",options) } end end def title new_title = nil if new_title @title = new_title end Lolita::Utils.dynamic_string(@title, :default => @type.to_s.humanize) end private def my_type self.class.to_s.split("::").last.downcase.to_sym end def set_default_attributes @name = "tab_#{self.__id__}" unless @name @title = set_default_title unless @title end def set_default_title if defined?(::I18n) translation_str = "lolita.tabs.titles.#{@type}" Proc.new{ ::I18n.t(translation_str) } else @type.to_s.humanize end end def get_class(association_name) if association_name.is_a?(Symbol) && assoc = self.current_dbi.reflect_on_association(association_name) if dbi.nested_attributes_options[association_name].nil? && dbi.nested_attributes_options[association_name.to_s].nil? raise ArgumentError, "#{dbi.klass} association named `#{association_name}` called nested_fields_XXX without accepts_nested_attributes_for :#{association_name}" end assoc.klass else raise ArgumentError, "Association named `#{association_name}` not found for #{self.current_dbi.klass}." end end def builder_local_variable_name :tab end end end end end