Module: Dom::Node

Included in:
CodeObject::Base, Document::Document, NoDoc
Defined in:
lib/dom/node.rb

Overview

Note:

including Class should implement instance-variable @name

Node can be seen as an aspect or feature of another Object. Therefore it can be mixed in to add node-functionality to a class. Such functionality is used by code-objects and documents.

Instance Variables


The following instance-variables will be set while including Dom::Node into your class:

  • ‘@name` (should be already set in your including class)

  • ‘@parent`

  • ‘@children`

Examples:

class MyObject
  include Dom::Node

  # The contructor of MyObject should call init_node
  def initialize(name)
    super
    @name = name
  end
end

# MyObject can now be used as a node within our Domtree
@baz = MyObject.new 'Baz'
@baz.add_node(MyObject.new 'foo')
@baz.add_node(MyObject.new 'bar')

# These Nodes get inserted into Dom, so it looks like
Dom.print_tree
#=>
#-Baz
#  -foo
#  -bar

See Also:

Constant Summary collapse

NS_SEP_STRING =
'.'
NS_SEP =
/#{Regexp.escape(NS_SEP_STRING)}/
NODENAME =
/[0-9a-zA-Z$_]+/
LEAF =
/^#{NS_SEP}(?<name>#{NODENAME})$/
ABSOLUTE =
/^(?<first>#{NODENAME})(?<rest>(#{NS_SEP}#{NODENAME})*)$/
RELATIVE =
/^#{NS_SEP}(?<first>#{NODENAME})(?<rest>(#{NS_SEP}#{NODENAME})*)$/

Initialization collapse

Traversing collapse

Manipulation collapse

Name- and Path-Handling collapse

Instance Method Details

#[](childname) ⇒ Dom::Node #[](path) ⇒ Dom::Node

Get’s the child of this node, which is identified by ‘path`

Examples:

childname

Dom[:Core]             #=> #<CodeObject::Object:Core @parent=__ROOT__ …>
Dom['Core']            #=> #<CodeObject::Object:Core @parent=__ROOT__ …>

path

Dom['Core.logger.log'] #=> #<CodeObject::Function:log @parent=logger …>

Overloads:



124
125
126
127
128
129
130
131
132
133
134
135
# File 'lib/dom/node.rb', line 124

def [](path)
  return @children[path] if path.is_a? Symbol
  return @children[path.to_sym] if path.split(NS_SEP_STRING).size == 1
  
  path = path.split(NS_SEP_STRING)
  child = @children[path.shift.to_sym]
  
  return nil if child.nil?
  
  # decend recursive
  child[path.join(NS_SEP_STRING)]
end

#add_node(path, node) ⇒ Object #add_node(node) ⇒ Object

There are three different cases

1. Last Childnode i.e. ".foo"
   -> Append node as child
2. absolute path i.e. "Foo"
   -> delegate path resolution to Dom.root
3. relative path i.e. ".bar.foo"
      a. if there is a matching child for first element, delegate
         adding to this node
      b. create NoDoc node and delegate rest to this NoDoc

Overloads:

  • #add_node(path, node) ⇒ Object

    Parameters:

  • #add_node(node) ⇒ Object

    Parameters:



259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
# File 'lib/dom/node.rb', line 259

def add_node(*args)      
  
  # Grabbing right overload
  if args.count == 2
    node = args[1]
    path = args[0]        
  elsif args.count == 1
    node = args[0]
            
    path = '.'+node.name
    raise NoPathGiven.new("#{node} has no path.") if path.nil?
  else
    raise ArgumentError.new "Wrong number of arguments #{args.count} for 1 or 2"      
  end      
  
  leaf = LEAF.match path
  
  if leaf
    # node found, lets insert it
    add_child(leaf[:name], node)
    
  else
    matches = RELATIVE.match path
    
    if matches
      name = matches[:first].to_s
      
      # node not found, what to do?
      add_child(name, NoDoc.new(name)) if self[name].nil?
      
      # everything fixed, continue with rest
      self[name].add_node(matches[:rest], node)          
  
    else
      # has to be an absolute path or something totally strange
      raise WrongPath.new(path) unless ABSOLUTE.match path
      
      # begin at top, if absolute
      Dom.add_node '.' + path, node        
    end        
  end      
end

#childrenHash<Symbol, Dom::Node>

Returns all immediately associated children of this ‘node`

Returns:



103
104
105
# File 'lib/dom/node.rb', line 103

def children
  @children
end

#each_child(&block) ⇒ Object

Iterates recursivly over all children of this node and applies ‘block` to them

Parameters:

  • block (Block)


231
232
233
234
235
236
# File 'lib/dom/node.rb', line 231

def each_child(&block)
  @children.values.each do |child| 
    yield(child)
    child.each_child(&block) 
  end
end

#find(path) ⇒ Object

Alias for bracket-access

See Also:



139
140
141
# File 'lib/dom/node.rb', line 139

def find(path)
  self[path]
end

#has_children?Boolean

Returns if the current node is a leaf or not.

Returns:

  • (Boolean)


147
148
149
# File 'lib/dom/node.rb', line 147

def has_children?
  not (@children.nil? or @children.size == 0)
end

#initializeObject

The ‘constructor’ of Dom::Node. It initializes all required instance-variables (see above).



60
61
62
63
64
# File 'lib/dom/node.rb', line 60

def initialize
  super
  @children, @parent = {}, nil
  @name ||= "" # should be written by including class
end

#namespaceString

Like #qualified_name it finds the absolute path. But in contrast to qualified_name it will not include the current node.

Examples:

my_node.qualified_name #=> "Core.logger.log"
my_node.namespace #=> "Core.logger"

Returns:

See Also:



313
314
315
# File 'lib/dom/node.rb', line 313

def namespace
  parents.map{|p| p.name}.join NS_SEP_STRING
end

#parentDom::Node

‘node.parent` returns the parent node, if there is one. If no parent exists `nil` is returned. In this case `node` can be considered either as loose leave or root of the Dom.

Returns:

  • (Dom::Node)

    the parent node, if one exists



73
74
75
# File 'lib/dom/node.rb', line 73

def parent
  @parent
end

#parentsArray<Dom::Node>

searches all parents recursivly and returns an array, starting with the highest order parent (excluding the Dom.root) and ending with the

immediate parent.

Examples:

o1 = CodeObject::Base.new :foo
o2 = CodeObject::Base.new :bar, o1
o3 = CodeObject::Base.new :baz, o2

# no parents
o1.parents #=> []

# all parents
o3.parents #=> [#<CodeObject::Base:foo>, #<CodeObject::Base:bar>]

Returns:

  • (Array<Dom::Node>)

    the associated parents



94
95
96
97
# File 'lib/dom/node.rb', line 94

def parents
  return [] if @parent.nil? or @parent.parent.nil?
  @parent.parents << @parent
end

Generates a text-output on console, representing the subtree-structure of the current node. The current node is not being printed.

Examples:

example output

-Core
  -extend
  -extensions
  -logger
    -log
  -properties

Returns:

  • (nil)


342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
# File 'lib/dom/node.rb', line 342

def print_tree

  # Because parent = nil and parent = Dom.root are handled the same at #parents
  # we need to make a difference here
  if @parent.nil?
    level = 0
  else
    level = parents.count + 1
  end
  
  @children.each do |name, child|
    puts "#{"  " * level}-#{name.to_s}#{' (NoDoc)' if child.is_a? NoDoc}"
    child.print_tree
  end
  nil
end

#qualified_nameString

The **Qualified Name** equals the **Absolute Path** starting from the root-node of the Dom.

Examples:

my_node.qualified_name #=> "Core.logger.log"

Returns:



325
326
327
# File 'lib/dom/node.rb', line 325

def qualified_name
  parents.push(self).map{|p| p.name}.join NS_SEP_STRING
end

#resolve(nodename) ⇒ Dom::Node?

Resolves ‘nodename` in the current context and tries to find a matching Dom::Node. If `nodename` is not the current name and further cannot be found in the list of children the parent will be asked for resolution.

Given the following example, each query (except ‘.log`) can be resolved without ambiguity.

# -Core
#   -extend
#   -extensions
#   -logger
#     -log
#   -properties
#   -log

Dom[:Core][:logger].resolve '.log'
#=>  #<CodeObject::Function:log @parent=logger @children=[]> 

Dom[:Core][:logger][:log].resolve '.log'
#=>  #<CodeObject::Function:log @parent=logger @children=[]> 

Dom[:Core][:logger][:log].resolve '.logger'
#=>  #<CodeObject::Object:logger @parent=Core @children=[]> 

Dom[:Core].resolve '.log'
#=>  #<CodeObject::Function:log @parent=Core @children=[]> 

Dom[:Core][:properties].resolve '.log'
#=> #<CodeObject::Function:extend @parent=Core @children=[]> 

Dom[:Core][:logger].resolve 'foo'
#=> nil

Parameters:

Returns:



196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
# File 'lib/dom/node.rb', line 196

def resolve(nodename)
  
  return self if nodename == @name
  return nil if @children.nil? and @parent.nil?
  
  path = RELATIVE.match(nodename)
  
  if path        
    first, rest = path.captures
    
    # we did find the first part in our list of children
    if not @children.nil? and @children.has_key? first.to_sym
      
      # we have to continue our search  
      if rest != ""          
        @children[first.to_sym].resolve rest
      
      # Finish
      else
        @children[first.to_sym]
      end
    
    else
      @parent.resolve nodename unless @parent.nil?
    end
    
  # It's absolute?
  elsif ABSOLUTE.match nodename
    Dom.root.find nodename
  end
end

#siblingsArray<Dom::Node>?

Finds all siblings of the current node. If there is no parent, it will return ‘nil`.

Returns:



155
156
157
158
# File 'lib/dom/node.rb', line 155

def siblings
  return nil if parent.nil?
  @parent.children
end