require 'hooks/inheritable_attribute' require 'representable/definition' require 'representable/nokogiri_extensions' module Representable def self.included(base) base.class_eval do extend ClassMethods::Accessors, ClassMethods::Declarations extend Hooks::InheritableAttribute inheritable_attr :representable_attrs self.representable_attrs = [] inheritable_attr :explicit_representation_name # FIXME: move to Accessors. end end # Reads values from +doc+ and sets properties accordingly. def update_properties_from(doc) self.class.representable_bindings.each do |ref| next if block_given? and not yield ref # skip if block is false. # DISCUSS: will we keep that? value = ref.read(doc) send(ref.definition.setter, value) end self end private # Compiles the document going through all properties. def create_representation_with(doc) self.class.representable_bindings.each do |ref| next if block_given? and not yield ref # skip if block is false. # DISCUSS: will we keep that? value = public_send(ref.definition.getter) # DISCUSS: eventually move back to Ref. ref.write(doc, value) if value end doc end module ClassMethods # :nodoc: module Declarations def definition_class Definition end # Returns bindings for all properties. def representable_bindings representable_attrs.map {|attr| binding_for_definition(attr) } end # Declares a reference to a certain xml element, whether an attribute, a node, # or a typed collection of nodes. This method does not add a corresponding accessor # to the object. For that behavior see the similar methods: .xml_reader and .xml_accessor. # # == Sym Option # [sym] Symbol representing the name of the accessor. # # === Default naming # This name will be the default node or attribute name searched for, # if no other is declared. For example, # # xml_reader :bob # xml_accessor :pony, :from => :attr # # are equivalent to: # # xml_reader :bob, :from => 'bob' # xml_accessor :pony, :from => '@pony' # # == Options # === :as # ==== Basic Types # Allows you to specify one of several basic types to return the value as. For example # # xml_reader :count, :as => Integer # # is equivalent to: # # xml_reader(:count) {|val| Integer(val) unless val.empty? } # # Such block shorthands for Integer, Float, Fixnum, BigDecimal, Date, Time, and DateTime # are currently available, but only for non-Hash declarations. # # To reference many elements, put the desired type in a literal array. e.g.: # # xml_reader :counts, :as => [Integer] # # Even an array of text nodes can be specified with :as => [] # # xml_reader :quotes, :as => [] # # === Other ROXML Class # Declares an accessor that represents another ROXML class as child XML element # (one-to-one or composition) or array of child elements (one-to-many or # aggregation) of this type. Default is one-to-one. For one-to-many, simply pass the class # as the only element in an array. # # Composition example: # # # Pragmatic Bookshelf # # # # Can be mapped using the following code: # class Book # xml_reader :publisher, :as => Publisher # end # # Aggregation example: # # # # # # # # Can be mapped using the following code: # class Library # xml_reader :books, :as => [Book], :in => "books" # end # # If you don't have the tag to wrap around the list of tags: # # Ruby books # # # # # You can skip the wrapper argument: # xml_reader :books, :as => [Book] # # === :from # The name by which the xml value will be found, either an attribute or tag name in XML. # Default is sym, or the singular form of sym, in the case of arrays and hashes. # # This value may also include XPath notation. # # ==== :from => :content # When :from is set to :content, this refers to the content of the current node, # rather than a sub-node. It is equivalent to :from => '.' # # Example: # class Contributor # xml_reader :name, :from => :content # xml_reader :role, :from => :attr # end # # To map: # James Wick # # ==== :from => :attr # When :from is set to :attr, this refers to the content of an attribute, # rather than a sub-node. It is equivalent to :from => '@attribute_name' # # Example: # class Book # xml_reader :isbn, :from => "@ISBN" # xml_accessor :title, :from => :attr # :from defaults to '@title' # end # # To map: # # # ==== :from => :text # The default source, if none is specified, this means the accessor # represents a text node from XML. This is documented for completeness # only. You should just leave this option off when you want the default behavior, # as in the examples below. # # :text is equivalent to :from => accessor_name, and you should specify the # actual node name (and, optionally, a namespace) if it differs, as in the case of :author below. # # Example: # class Book # xml_reader :author, :from => 'Author' # xml_accessor :description, :cdata => true # xml_reader :title # end # # To map: # # Programming Ruby: the pragmatic programmers' guide # # David Thomas # # # Likewise, a number of :text node values can be collected in an array like so: # # Example: # class Library # xml_reader :books, :as => [] # end # # To map: # # To kill a mockingbird # House of Leaves # Gödel, Escher, Bach # # # === Other Options # [:in] An optional name of a wrapping tag for this XML accessor. # This can include other xpath values, which will be joined with :from with a '/' # [:required] If true, throws RequiredElementMissing when the element isn't present # [:cdata] true for values which should be input from or output as cdata elements # [:to_xml] this proc is applied to the attributes value outputting the instance via #to_xml # def representable_property(*args) # TODO: make it accept 1-n props. attr = add_representable_property(*args) attr_reader(attr.getter) attr_writer(attr.getter) end def representable_collection(name, options={}) options[:as] = [options[:as]].compact representable_property(name, options) end private def add_representable_property(*args) definition_class.new(*args).tap do |attr| representable_attrs << attr end end end module Accessors def representation_name=(name) self.explicit_representation_name = name end def representation_name explicit_representation_name or name.split('::').last. gsub(/([A-Z]+)([A-Z][a-z])/,'\1_\2'). gsub(/([a-z\d])([A-Z])/,'\1_\2'). downcase end end end end