Parent

Class/Module Index [+]

Quicksearch

RGen::Fragment::ModelFragment

A model fragment is a list of root model elements associated with a location (e.g. a file)

Optionally, an arbitrary data object may be associated with the fragment. The data object will also be stored in the cache.

If an element within the fragment changes this must be indicated to the fragment by calling mark_changed.

Attributes

data[RW]
fragment_ref[RW]
location[RW]
root_elements[R]

Public Class Methods

new(location, options={}) click to toggle source

Create a model fragment

:data
  data object associated with this fragment

:identifier_provider
  identifier provider to be used when resolving references
  it must be a proc which receives a model element and must return 
  that element's identifier or nil if the element has no identifier
# File lib/rgen/fragment/model_fragment.rb, line 49
def initialize(location, options={})
  @location = location
  @fragment_ref = FragmentRef.new
  @fragment_ref.fragment = self
  @data = options[:data]
  @resolved_refs = nil 
  @changed = false
  @identifier_provider = options[:identifier_provider]
end

Public Instance Methods

build_index() click to toggle source

Builds the index of all elements within this fragment having an identifier the index is an array of 2-element arrays holding the identifier and the element

# File lib/rgen/fragment/model_fragment.rb, line 133
def build_index
  raise "cannot build index without an identifier provider" unless @identifier_provider
  @index = elements.collect { |e|
    ident = @identifier_provider.call(e, nil)
    ident && !ident.empty? ? [ident, e] : nil 
  }.compact
end
changed?() click to toggle source

Indicates whether the fragment has been changed or not

# File lib/rgen/fragment/model_fragment.rb, line 92
def changed?
  @changed
end
elements() click to toggle source

Returns all elements within this fragment

# File lib/rgen/fragment/model_fragment.rb, line 98
def elements
  return @elements if @elements
  @elements = []
  @root_elements.each do |e|
    all_child_elements(e, @elements)
  end
  @elements
end
index() click to toggle source

Returns the index of the element contained in this fragment.

# File lib/rgen/fragment/model_fragment.rb, line 109
def index
  build_index unless @index
  @index
end
mark_changed() click to toggle source

Must be called when any of the elements in this fragment has been changed

# File lib/rgen/fragment/model_fragment.rb, line 76
def mark_changed
  @changed = true
  @elements = nil
  @index = nil
  @unresolved_refs = nil
  @resolved_refs = :dirty 
end
mark_unchanged() click to toggle source

Can be used to reset the change status to unchanged.

# File lib/rgen/fragment/model_fragment.rb, line 86
def mark_unchanged
  @changed = false
end
resolve_external(external_index, options) click to toggle source

Resolves references to external fragments using the external_index provided. The external index must be a Hash mapping identifiers uniquely to model elements.

Options:

:fragment_provider: 
  If a +fragment_provider+ is given, the resolve step can be reverted later on 
  by a call to unresolve_external or unresolve_external_fragment. The fragment provider
  is a proc which receives a model element and must return the fragment in which it is
  contained.

:use_target_type: 
  reference resolver uses the expected target type to narrow the set of possible targets
# File lib/rgen/fragment/model_fragment.rb, line 171
def resolve_external(external_index, options)
  fragment_provider = options[:fragment_provider]
  resolver = RGen::Instantiator::ReferenceResolver.new(
    :identifier_resolver => proc {|ident| external_index[ident] })
  if fragment_provider
    @resolved_refs = {} if @resolved_refs.nil? || @resolved_refs == :dirty
    on_resolve = proc { |ur, target|
      target_fragment = fragment_provider.call(target)
      target_fragment ||= :unknown
      raise "can not resolve local reference in resolve_external, call resolve_local first"            if target_fragment == self
      @resolved_refs[target_fragment] ||= []
      @resolved_refs[target_fragment] << ResolvedReference.new(ur, target)
    } 
    @unresolved_refs = resolver.resolve(unresolved_refs, :on_resolve => on_resolve, :use_target_type => options[:use_target_type])
  else
    @unresolved_refs = resolver.resolve(unresolved_refs, :use_target_type => options[:use_target_type])
  end
end
resolve_local(options={}) click to toggle source

Resolves local references (within this fragment) as far as possible

Options:

:use_target_type: 
  reference resolver uses the expected target type to narrow the set of possible targets
# File lib/rgen/fragment/model_fragment.rb, line 148
def resolve_local(options={})
  resolver = RGen::Instantiator::ReferenceResolver.new
  index.each do |i|
    resolver.add_identifier(i[0], i[1])
  end
  @unresolved_refs = resolver.resolve(unresolved_refs, :use_target_type => options[:use_target_type])
end
set_root_elements(root_elements, options={}) click to toggle source

Set the root elements, normally done by an instantiator.

For optimization reasons the instantiator of the fragment may provide data explicitly which is normally derived by the fragment itself. In this case it is essential that this data is consistent with the fragment.

# File lib/rgen/fragment/model_fragment.rb, line 65
def set_root_elements(root_elements, options={})
  @root_elements = root_elements 
  @elements = options[:elements]
  @index = options[:index]
  @unresolved_refs = options[:unresolved_refs]
  @resolved_refs = nil 
  @changed = false
end
unresolve_external() click to toggle source

Unresolve outgoing references to all external fragments, i.e. references which used to be represented by an unresolved reference from within this fragment. Note, that there may be more references to external fragments due to references which were represented by unresolved references from within other fragments.

# File lib/rgen/fragment/model_fragment.rb, line 196
def unresolve_external
  return if @resolved_refs.nil?
  raise "can not unresolve, missing fragment information" if @resolved_refs == :dirty || @resolved_refs[:unknown]
  rrefs = @resolved_refs.values.flatten
  @resolved_refs = {}
  unresolve_refs(rrefs)
end
unresolve_external_fragment(fragment) click to toggle source

Like unresolve_external but only unresolve references to external fragment fragment

# File lib/rgen/fragment/model_fragment.rb, line 206
def unresolve_external_fragment(fragment)
  return if @resolved_refs.nil?
  raise "can not unresolve, missing fragment information" if @resolved_refs == :dirty || @resolved_refs[:unknown]
  rrefs = @resolved_refs[fragment]
  @resolved_refs.delete(fragment)
  unresolve_refs(rrefs) if rrefs
end
unresolved_refs() click to toggle source

Returns all unresolved references within this fragment, i.e. references to MMProxy objects

# File lib/rgen/fragment/model_fragment.rb, line 116
def unresolved_refs
  return @unresolved_refs if @unresolved_refs
  @unresolved_refs = []
  elements.each do |e|
    each_reference_target(e) do |r, t|
      if t.is_a?(RGen::MetamodelBuilder::MMProxy)
        @unresolved_refs << 
          RGen::Instantiator::ReferenceResolver::UnresolvedReference.new(e, r.name, t)
      end
    end
  end
  @unresolved_refs
end

[Validate]

Generated with the Darkfish Rdoc Generator 2.