Mercurial > pub > Impl
comparison lib/IMPL/DOM/Navigator/Builder.pm @ 407:c6e90e02dd17 ref20150831
renamed Lib->lib
| author | cin |
|---|---|
| date | Fri, 04 Sep 2015 19:40:23 +0300 |
| parents | |
| children |
comparison
equal
deleted
inserted
replaced
| 406:f23fcb19d3c1 | 407:c6e90e02dd17 |
|---|---|
| 1 package IMPL::DOM::Navigator::Builder; | |
| 2 use strict; | |
| 3 use warnings; | |
| 4 | |
| 5 use IMPL::Const qw(:prop); | |
| 6 | |
| 7 use parent qw(IMPL::DOM::Navigator); | |
| 8 use IMPL::Class::Property; | |
| 9 require IMPL::DOM::Navigator::SchemaNavigator; | |
| 10 require IMPL::DOM::Schema::ValidationError; | |
| 11 use IMPL::DOM::Document; | |
| 12 | |
| 13 BEGIN { | |
| 14 private _direct property _schemaNavi => PROP_RW; | |
| 15 private _direct property _docClass => PROP_RW; | |
| 16 public _direct property Document => PROP_RO; | |
| 17 public _direct property ignoreUndefined => PROP_RO; | |
| 18 } | |
| 19 | |
| 20 our %CTOR = ( | |
| 21 'IMPL::DOM::Navigator' => sub { IMPL::DOM::Document->Empty; } | |
| 22 ); | |
| 23 | |
| 24 sub CTOR { | |
| 25 my ($this,$docClass,$schema,%opts) = @_; | |
| 26 | |
| 27 $this->{$_docClass} = $docClass; | |
| 28 $this->{$_schemaNavi} = $schema ? IMPL::DOM::Navigator::SchemaNavigator->new($schema) : undef; | |
| 29 | |
| 30 $this->{$ignoreUndefined} = $opts{ignoreUndefined} if $opts{ignoreUndefined}; | |
| 31 } | |
| 32 | |
| 33 sub NavigateCreate { | |
| 34 my ($this,$nodeName,%props) = @_; | |
| 35 | |
| 36 if (my $schemaType = $this->{$_schemaNavi}->NavigateName($nodeName)) { | |
| 37 my $class = $schemaType->can('nativeType') ? $schemaType->nativeType || 'IMPL::DOM::Node' : 'IMPL::DOM::Node'; | |
| 38 | |
| 39 my $schemaNode = $this->{$_schemaNavi}->SourceSchemaNode; | |
| 40 | |
| 41 $props{schemaType} = $schemaType; | |
| 42 $props{schemaNode} = $schemaNode; | |
| 43 | |
| 44 my $node; | |
| 45 if (! $this->{$Document}) { | |
| 46 # keep reference to the schema document | |
| 47 $props{schemaDocument} = $this->{$_schemaNavi}->schema; | |
| 48 $node = $this->{$Document} = $this->{$_docClass}->new(nodeName => $nodeName,%props); | |
| 49 $this->_initNavigator($node); | |
| 50 } else { | |
| 51 die new IMPL::InvalidOperationException('Can\'t create a second top level element') unless $this->Current; | |
| 52 $node = $this->{$Document}->Create($nodeName,$class,\%props); | |
| 53 $this->Current->appendChild($node); | |
| 54 $this->internalNavigateNodeSet($node); | |
| 55 } | |
| 56 | |
| 57 return $node; | |
| 58 } else { | |
| 59 die new IMPL::InvalidOperationException("The specified node is undefined", $nodeName) | |
| 60 if !$this->ignoreUndefined; | |
| 61 return; | |
| 62 } | |
| 63 } | |
| 64 | |
| 65 sub Back { | |
| 66 my ($this) = @_; | |
| 67 | |
| 68 $this->{$_schemaNavi}->SchemaBack(); | |
| 69 $this->SUPER::Back(); | |
| 70 } | |
| 71 | |
| 72 sub saveState { | |
| 73 my ($this) = @_; | |
| 74 | |
| 75 $this->{$_schemaNavi}->saveState; | |
| 76 $this->SUPER::saveState; | |
| 77 } | |
| 78 | |
| 79 sub restoreState { | |
| 80 my ($this) = @_; | |
| 81 | |
| 82 $this->{$_schemaNavi}->restoreState; | |
| 83 $this->SUPER::restoreState; | |
| 84 } | |
| 85 | |
| 86 sub document { | |
| 87 goto &Document; | |
| 88 } | |
| 89 | |
| 90 1; | |
| 91 | |
| 92 __END__ | |
| 93 | |
| 94 =pod | |
| 95 | |
| 96 =head1 NAME | |
| 97 | |
| 98 C< IMPL::DOM::Navigator::Builder > - Навигатор, строящий документ по указанной схеме. | |
| 99 | |
| 100 =head1 SYNOPSIS | |
| 101 | |
| 102 =begin code | |
| 103 | |
| 104 my $builder = new IMPL::DOM::Navigator::Builder(new MyApp::Document,$schema); | |
| 105 my $reader = new IMPL::DOM::XMLReader(Navigator => $builder); | |
| 106 | |
| 107 $reader->ParseFile("document.xml"); | |
| 108 | |
| 109 my @errors = $schema->Validate($builder->Document); | |
| 110 | |
| 111 =end code | |
| 112 | |
| 113 =head1 DESCRIPTION | |
| 114 | |
| 115 Построитель DOM документов по указанной схеме. Обычно используется в связке | |
| 116 с объектами для чтения такими как C<IMPL::DOM::XMLReader>. | |
| 117 | |
| 118 =head1 MEMBERS | |
| 119 | |
| 120 =head2 C< CTOR($classDocument,$schema, %opts) > | |
| 121 | |
| 122 Создает новый объект, принимает на вход класс документа (или фабрику, например | |
| 123 L<IMPL::Object::Factory>) и схему. В процессе процедуры построения документа | |
| 124 будет создан объект документа. | |
| 125 | |
| 126 Необязательные именованные параметры | |
| 127 | |
| 128 =over | |
| 129 | |
| 130 =item C<ignoreUndefined> | |
| 131 | |
| 132 C<NavigateCreate> не будет вызывать исключение, если запрашиваемый узел не | |
| 133 найден в схеме, но будет возвращать C<undef>. | |
| 134 | |
| 135 =back | |
| 136 | |
| 137 =head2 C< NavigateCreate($nodeName,%props) > | |
| 138 | |
| 139 Создает новый узел с указанным именем и переходит в него. В случае если в схеме | |
| 140 подходящий узел не найден, то вызывается исключение или будет возвращено | |
| 141 C<undef> см. C<ignoreUndefined>. | |
| 142 | |
| 143 При этом по имени узла ищется его схема, после чего определяется класс для | |
| 144 создания экземпляра узла и созданный узел доавляется в документ. При создании | |
| 145 нового узла используется метод документа C<< IMPL::DOM::Document->Create >> | |
| 146 | |
| 147 Свойства узла передаются при создании через параметр C<%props>, но имя | |
| 148 создаваемого узла НЕ может быть переопределено свойством C<nodeName>, оно будет | |
| 149 проигнорировано. | |
| 150 | |
| 151 Свойства узла будут преобразованы при помощи заданных в схеме заполнителей | |
| 152 C<inflator>. | |
| 153 | |
| 154 =head2 C<[get]document > | |
| 155 | |
| 156 Свойство, которое содержит документ по окончании процедуры построения. | |
| 157 | |
| 158 =head2 C<[get]buildErrors> | |
| 159 | |
| 160 Ошибки, возникшие в процессе построения документа. | |
| 161 | |
| 162 =head2 C<[get]ignoreUndefined> | |
| 163 | |
| 164 Опция, заданная при создании построителя, отвечающая за обработку узлов | |
| 165 не найденных в схеме. | |
| 166 | |
| 167 =cut |
