Class: REXMLUtilityNode

Inherits:
Object show all
Defined in:
lib/core_extensions.rb

Overview

This is a slighly modified version of the XMLUtilityNode from merb.devjavu.com/projects/merb/ticket/95 ([email protected]) It’s mainly just adding vowels, as I ht cd wth n vwls :) This represents the hard part of the work, all I did was change the underlying parser.

Instance Attribute Summary collapse

Class Method Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(name, attributes = {}) ⇒ REXMLUtilityNode

Returns a new instance of REXMLUtilityNode.



128
129
130
131
132
133
134
135
136
137
# File 'lib/core_extensions.rb', line 128

def initialize(name, attributes = {})
  @name         = name.tr("-", "_")
  # leave the type alone if we don't know what it is
  @type         = self.class.available_typecasts.include?(attributes["type"]) ? attributes.delete("type") : attributes["type"]

  @nil_element  = attributes.delete("nil") == "true"
  @attributes   = undasherize_keys(attributes)
  @children     = []
  @text         = false
end

Instance Attribute Details

#attributesObject

Returns the value of attribute attributes.



94
95
96
# File 'lib/core_extensions.rb', line 94

def attributes
  @attributes
end

#childrenObject

Returns the value of attribute children.



94
95
96
# File 'lib/core_extensions.rb', line 94

def children
  @children
end

#nameObject

Returns the value of attribute name.



94
95
96
# File 'lib/core_extensions.rb', line 94

def name
  @name
end

#typeObject

Returns the value of attribute type.



94
95
96
# File 'lib/core_extensions.rb', line 94

def type
  @type
end

Class Method Details

.available_typecastsObject



104
105
106
# File 'lib/core_extensions.rb', line 104

def self.available_typecasts
  @@available_typecasts
end

.available_typecasts=(obj) ⇒ Object



108
109
110
# File 'lib/core_extensions.rb', line 108

def self.available_typecasts=(obj)
  @@available_typecasts = obj
end

.typecastsObject



96
97
98
# File 'lib/core_extensions.rb', line 96

def self.typecasts
  @@typecasts
end

.typecasts=(obj) ⇒ Object



100
101
102
# File 'lib/core_extensions.rb', line 100

def self.typecasts=(obj)
  @@typecasts = obj
end

Instance Method Details

#add_node(node) ⇒ Object



139
140
141
142
# File 'lib/core_extensions.rb', line 139

def add_node(node)
  @text = true if node.is_a? String
  @children << node
end

#inner_htmlObject

Get the inner_html of the REXML node.



245
246
247
# File 'lib/core_extensions.rb', line 245

def inner_html
  @children.join
end

#to_hashObject



144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
# File 'lib/core_extensions.rb', line 144

def to_hash
  if @type == "file"
    f = StringIO.new((@children.first || '').unpack('m').first)
    class << f
      attr_accessor :original_filename, :content_type
    end
    f.original_filename = attributes['name'] || 'untitled'
    f.content_type = attributes['content_type'] || 'application/octet-stream'
    return {name => f}
  end

  if @text
    return { name => typecast_value( translate_xml_entities( inner_html ) ) }
  else
    #change repeating groups into an array
    groups = @children.inject({}) { |s,e| (s[e.name] ||= []) << e; s }

    out = nil
    if @type == "array"
      out = []
      groups.each do |k, v|
        if v.size == 1
          out << v.first.to_hash.entries.first.last
        else
          out << v.map{|e| e.to_hash[k]}
        end
      end
      out = out.flatten

    else # If Hash
      out = {}
      groups.each do |k,v|
        if v.size == 1
          out.merge!(v.first)
        else
          out.merge!( k => v.map{|e| e.to_hash[k]})
        end
      end
      out.merge! attributes unless attributes.empty?
      out = out.empty? ? nil : out
    end

    if @type && out.nil?
      { name => typecast_value(out) }
    else
      { name => out }
    end
  end
end

#to_htmlString

Converts the node into a readable HTML node.

Returns:

  • (String)

    The HTML node in text form.



252
253
254
255
# File 'lib/core_extensions.rb', line 252

def to_html
  attributes.merge!(:type => @type ) if @type
  "<#{name}#{attributes.to_xml_attributes}>#{@nil_element ? '' : inner_html}</#{name}>"
end

#to_sObject



258
259
260
# File 'lib/core_extensions.rb', line 258

def to_s
  to_html
end

#translate_xml_entities(value) ⇒ #gsub

Convert basic XML entities into their literal values.

Parameters:

  • value (#gsub)

    An XML fragment.

Returns:

  • (#gsub)

    The XML fragment after converting entities.



228
229
230
231
232
233
234
# File 'lib/core_extensions.rb', line 228

def translate_xml_entities(value)
  value.gsub(/&lt;/,   "<").
        gsub(/&gt;/,   ">").
        gsub(/&quot;/, '"').
        gsub(/&apos;/, "'").
        gsub(/&amp;/,  "&")
end

#typecast_value(value) ⇒ Integer, ...

Note:

If self does not have a “type” key, or if it’s not one of the options specified above, the raw value will be returned.

Typecasts a value based upon its type. For instance, if node has #type == “integer”, #=> 12]}

Parameters:

  • value (String)

    The value that is being typecast.

Returns:



217
218
219
220
221
# File 'lib/core_extensions.rb', line 217

def typecast_value(value)
  return value unless @type
  proc = self.class.typecasts[@type]
  proc.nil? ? value : proc.call(value)
end

#undasherize_keys(params) ⇒ Object

Take keys of the form foo-bar and convert them to foo_bar



237
238
239
240
241
242
# File 'lib/core_extensions.rb', line 237

def undasherize_keys(params)
  params.keys.each do |key, value|
    params[key.tr("-", "_")] = params.delete(key)
  end
  params
end