blib/lib/XML/Handler/ExtOn.pm | |||
---|---|---|---|
Criterion | Covered | Total | % |
statement | 31 | 33 | 93.9 |
branch | n/a | ||
condition | n/a | ||
subroutine | 11 | 11 | 100.0 |
pod | n/a | ||
total | 42 | 44 | 95.4 |
line | stmt | bran | cond | sub | pod | time | code |
---|---|---|---|---|---|---|---|
1 | package XML::Handler::ExtOn; | ||||||
2 | |||||||
3 | #$Id: ExtOn.pm 368 2008-11-24 09:55:03Z zag $ | ||||||
4 | |||||||
5 | =pod | ||||||
6 | |||||||
7 | =head1 NAME | ||||||
8 | |||||||
9 | XML::Handler::ExtOn - The handler for expansion of Perl SAX by objects. | ||||||
10 | |||||||
11 | =head1 SYNOPSYS | ||||||
12 | |||||||
13 | use XML::Handler::ExtOn; | ||||||
14 | |||||||
15 | For write XML: | ||||||
16 | |||||||
17 | use XML::Handler::ExtOn; | ||||||
18 | my $buf; | ||||||
19 | my $wrt = XML::SAX::Writer->new( Output => \$buf ); | ||||||
20 | my $ex_parser = new XML::Handler::ExtOn:: Handler => $wrt; | ||||||
21 | $ex_parser->start_document; | ||||||
22 | my $root = $ex_parser->mk_element("Root"); | ||||||
23 | $root->add_namespace( | ||||||
24 | "myns" => 'http://example.com/myns', | ||||||
25 | "myns_test", 'http://example.com/myns_test' | ||||||
26 | ); | ||||||
27 | $ex_parser->start_element( $root ); | ||||||
28 | my $el = $root->mk_element('vars'); | ||||||
29 | %{ $el->attrs_by_prefix("myns") } = ( v1 => 1, v2 => 3 ); | ||||||
30 | %{ $el->attrs_by_prefix("myns_test") } = | ||||||
31 | ( var1 => "test ns", var2 => "2333" ); | ||||||
32 | $root->add_content($el); | ||||||
33 | $ex_parser->end_element; | ||||||
34 | $ex_parser->end_document; | ||||||
35 | print $buf; | ||||||
36 | |||||||
37 | Result: | ||||||
38 | |||||||
39 | |||||||
40 | | ||||||
41 | xmlns:myns_test="http://example.com/myns_test"> | ||||||
42 | | ||||||
43 | myns_test:var1="test ns" | ||||||
44 | myns:v1="1" myns:v2="3"/> | ||||||
45 | |||||||
46 | |||||||
47 | For handle events | ||||||
48 | |||||||
49 | use base 'XML::Handler::ExtOn'; | ||||||
50 | |||||||
51 | Begin method for handle SAX event start_element: | ||||||
52 | |||||||
53 | sub on_start_element { | ||||||
54 | my ( $self, $elem ) = @_; | ||||||
55 | |||||||
56 | ... | ||||||
57 | |||||||
58 | Check localname for element and add tag C |
||||||
59 | |||||||
60 | if ( $elem->local_name eq 'gallery' ) { | ||||||
61 | $elem->add_content( | ||||||
62 | $self->mk_element('image')->add_content( | ||||||
63 | $self->mk_characters( "Image number: $_" ) | ||||||
64 | ) | ||||||
65 | ) for 1..2 ; | ||||||
66 | } | ||||||
67 | |||||||
68 | XML Before: | ||||||
69 | |||||||
70 | |||||||
71 | |
||||||
72 | |
||||||
73 | |||||||
74 | |||||||
75 | After: | ||||||
76 | |||||||
77 | |||||||
78 | |
||||||
79 | |
||||||
80 | |
||||||
81 | |
||||||
82 | |||||||
83 | |||||||
84 | |||||||
85 | Register namespace and set variables | ||||||
86 | |||||||
87 | $elem->add_namespace('demons','http://example.org/demo_namespace'); | ||||||
88 | $elem->add_namespace('ns2','http://example.org/ns2'); | ||||||
89 | #set attributes for name space | ||||||
90 | my $demo_attrs = $elem->attrs_by_prefix('demons'); | ||||||
91 | %{$demo_attrs} = ( variable1=>1, 'variable2'=>2); | ||||||
92 | #set attributes for namespace URI | ||||||
93 | my $ns2_attrs = $elem->attrs_by_ns_uri('http://example.org/ns2'); | ||||||
94 | %{$ns2_attrs} = ( var=> 'ns1', 'raw'=>2); | ||||||
95 | |||||||
96 | Result: | ||||||
97 | |||||||
98 | |||||||
99 | xmlns:ns2="http://example.org/ns2" | ||||||
100 | demons:variable2="2" ns2:var="ns1" | ||||||
101 | demons:variable1="1" ns2:raw="2"/> | ||||||
102 | |||||||
103 | Delete content of element | ||||||
104 | |||||||
105 | if ( $elem->local_name eq 'demo_delete') { | ||||||
106 | $elem->skip_content | ||||||
107 | } | ||||||
108 | |||||||
109 | XML before: | ||||||
110 | |||||||
111 | |||||||
112 | |
||||||
113 | |
||||||
114 | text |
||||||
115 | |||||||
116 | |||||||
117 | |||||||
118 | After: | ||||||
119 | |||||||
120 | |||||||
121 | |
||||||
122 | |
||||||
123 | |||||||
124 | |||||||
125 | Add XML: | ||||||
126 | |||||||
127 | $elem->add_content ( | ||||||
128 | $self->mk_from_xml(' text |
||||||
129 | ) | ||||||
130 | Can add element after current | ||||||
131 | |||||||
132 | ... | ||||||
133 | return [ $elem, $self->mk_element("after") ]; | ||||||
134 | } | ||||||
135 | |||||||
136 | =head1 DESCRIPTION | ||||||
137 | |||||||
138 | XML::Handler::ExtOn - SAX Handler designed for funny work with XML. It | ||||||
139 | provides an easy-to-use interface for XML applications by adding objects. | ||||||
140 | |||||||
141 | XML::Handler::ExtOn override some SAX events. Each time an SAX event starts, | ||||||
142 | a method by that name prefixed with `on_' is called with the B<"blessed"> | ||||||
143 | Element object to be processed. | ||||||
144 | |||||||
145 | XML::Handler::ExtOn implement the following methods: | ||||||
146 | |||||||
147 | =over | ||||||
148 | |||||||
149 | =item * on_start_document | ||||||
150 | |||||||
151 | =item * on_start_prefix_mapping | ||||||
152 | |||||||
153 | =item * on_start_element | ||||||
154 | |||||||
155 | =item * on_end_element | ||||||
156 | |||||||
157 | =item * on_characters | ||||||
158 | |||||||
159 | =item * on_cdata | ||||||
160 | |||||||
161 | =back | ||||||
162 | |||||||
163 | XML::Handler::ExtOn put all B |
||||||
164 | |||||||
165 | It compliant XML namespaces (http://www.w3.org/TR/REC-xml-names/), by support | ||||||
166 | I |
||||||
167 | |||||||
168 | XML::Handler::ExtOn provide methods for create XML, such as C |
||||||
169 | |||||||
170 | =head1 FUNCTIONS | ||||||
171 | |||||||
172 | =cut | ||||||
173 | |||||||
174 | 4 | 4 | 164692 | use strict; | |||
4 | 10 | ||||||
4 | 183 | ||||||
175 | 4 | 4 | 22 | use warnings; | |||
4 | 7 | ||||||
4 | 124 | ||||||
176 | |||||||
177 | 4 | 4 | 23 | use Carp; | |||
4 | 7 | ||||||
4 | 257 | ||||||
178 | 4 | 4 | 21 | use Data::Dumper; | |||
4 | 9 | ||||||
4 | 169 | ||||||
179 | |||||||
180 | 4 | 4 | 6236 | use XML::SAX::Base; | |||
4 | 98342 | ||||||
4 | 167 | ||||||
181 | 4 | 4 | 3224 | use XML::Handler::ExtOn::Element; | |||
4 | 12 | ||||||
4 | 112 | ||||||
182 | 4 | 4 | 2326 | use XML::Handler::ExtOn::Context; | |||
4 | 14 | ||||||
4 | 135 | ||||||
183 | 4 | 4 | 2839 | use XML::Handler::ExtOn::IncXML; | |||
4 | 10 | ||||||
4 | 118 | ||||||
184 | 4 | 4 | 3543 | use XML::Filter::SAX1toSAX2; | |||
4 | 18135 | ||||||
4 | 179 | ||||||
185 | 4 | 4 | 2600 | use XML::Handler::ExtOn::SAX12ExtOn; | |||
4 | 11 | ||||||
4 | 120 | ||||||
186 | 4 | 4 | 1842 | use XML::Parser::PerlSAX; | |||
0 | |||||||
0 | |||||||
187 | |||||||
188 | require Exporter; | ||||||
189 | *import = \&Exporter::import; | ||||||
190 | @XML::Handler::ExtOn::EXPORT_OK = qw( create_pipe ); | ||||||
191 | |||||||
192 | =head1 create_pipe "flt_n1",$some_handler, $out_handler | ||||||
193 | |||||||
194 | use last arg as handler for out. | ||||||
195 | |||||||
196 | return parser ref. | ||||||
197 | |||||||
198 | my $h1 = new MyHandler1::; | ||||||
199 | my $filter = create_pipe( 'MyHandler1', $h1 ); | ||||||
200 | $filter->parse(' TEST |
||||||
201 | |||||||
202 | =cut | ||||||
203 | |||||||
204 | sub create_pipe { | ||||||
205 | my @args = | ||||||
206 | reverse( "XML::Parser::PerlSAX", "XML::Handler::ExtOn::SAX12ExtOn", @_ ); | ||||||
207 | my $out_handler = shift @args; | ||||||
208 | foreach my $f (@args) { | ||||||
209 | unless ( ref($f) ) { | ||||||
210 | $out_handler = $f->new( Handler => $out_handler ); | ||||||
211 | } elsif ( UNIVERSAL::isa( $f, 'XML::SAX::Base')) { | ||||||
212 | $f->set_handler( $out_handler ); | ||||||
213 | $out_handler = $f | ||||||
214 | |||||||
215 | } | ||||||
216 | } | ||||||
217 | return $out_handler; | ||||||
218 | } | ||||||
219 | |||||||
220 | use base 'XML::SAX::Base'; | ||||||
221 | use vars qw( $AUTOLOAD); | ||||||
222 | $XML::Handler::ExtOn::VERSION = '0.06'; | ||||||
223 | ### install get/set accessors for this object. | ||||||
224 | for my $key (qw/ context _objects_stack _cdata_mode _cdata_characters/) { | ||||||
225 | no strict 'refs'; | ||||||
226 | *{ __PACKAGE__ . "::$key" } = sub { | ||||||
227 | my $self = shift; | ||||||
228 | $self->{___EXT_on_attrs}->{$key} = $_[0] if @_; | ||||||
229 | return $self->{___EXT_on_attrs}->{$key}; | ||||||
230 | } | ||||||
231 | } | ||||||
232 | |||||||
233 | =head1 METHODS | ||||||
234 | |||||||
235 | =cut | ||||||
236 | |||||||
237 | sub new { | ||||||
238 | my $class = shift; | ||||||
239 | my $self = &XML::SAX::Base::new( $class, @_, ); | ||||||
240 | $self->_objects_stack( [] ); | ||||||
241 | $self->_cdata_mode(0); | ||||||
242 | my $buf; | ||||||
243 | $self->_cdata_characters( \$buf ); #setup cdata buffer | ||||||
244 | my $doc_context = new XML::Handler::ExtOn::Context::; | ||||||
245 | $self->context($doc_context); | ||||||
246 | return $self; | ||||||
247 | } | ||||||
248 | |||||||
249 | =head2 on_start_document $document | ||||||
250 | |||||||
251 | Method handle C |
||||||
252 | variables. | ||||||
253 | |||||||
254 | sub on_start_document { | ||||||
255 | my $self = shift; | ||||||
256 | $self->{_LINKS_ARRAY} = []; | ||||||
257 | $self->SUPER::on_start_document(@_); | ||||||
258 | } | ||||||
259 | |||||||
260 | =cut | ||||||
261 | |||||||
262 | sub on_start_document { | ||||||
263 | my ( $self, $document ) = @_; | ||||||
264 | $self->SUPER::start_document($document); | ||||||
265 | } | ||||||
266 | |||||||
267 | sub start_document { | ||||||
268 | my ( $self, $document ) = @_; | ||||||
269 | return if $self->{___EXT_on_attrs}->{_skip_start_docs}++; | ||||||
270 | $self->on_start_document($document); | ||||||
271 | } | ||||||
272 | |||||||
273 | sub end_document { | ||||||
274 | my $self = shift; | ||||||
275 | my $var = --$self->{___EXT_on_attrs}->{_skip_start_docs}; | ||||||
276 | return if $var; | ||||||
277 | $self->SUPER::end_document(@_); | ||||||
278 | } | ||||||
279 | |||||||
280 | =head2 on_start_prefix_mapping prefix1=>ns_uri1[, prefix2=>ns_uri2] | ||||||
281 | |||||||
282 | Called on C |
||||||
283 | |||||||
284 | sub on_start_prefix_mapping { | ||||||
285 | my $self = shift; | ||||||
286 | my %map = @_; | ||||||
287 | $self->SUPER::start_prefix_mapping(@_) | ||||||
288 | } | ||||||
289 | |||||||
290 | =cut | ||||||
291 | |||||||
292 | sub on_start_prefix_mapping { | ||||||
293 | my $self = shift; | ||||||
294 | my %map = @_; | ||||||
295 | while ( my ( $pref, $ns_uri ) = each %map ) { | ||||||
296 | $self->add_namespace($pref, $ns_uri); | ||||||
297 | $self->SUPER::start_prefix_mapping( | ||||||
298 | { | ||||||
299 | Prefix => $pref, | ||||||
300 | NamespaceURI => $ns_uri | ||||||
301 | } | ||||||
302 | ); | ||||||
303 | } | ||||||
304 | } | ||||||
305 | |||||||
306 | # | ||||||
307 | # { Prefix => 'xlink', NamespaceURI => 'http://www.w3.org/1999/xlink' } | ||||||
308 | # | ||||||
309 | |||||||
310 | sub start_prefix_mapping { | ||||||
311 | my $self = shift; | ||||||
312 | |||||||
313 | #declare namespace for current context | ||||||
314 | # my $context = $self->context; | ||||||
315 | # if ( my $current = $self->current_element ) { | ||||||
316 | # $context = $current->ns; | ||||||
317 | # } | ||||||
318 | my %map = (); | ||||||
319 | foreach my $ref (@_) { | ||||||
320 | my ( $prefix, $ns_uri ) = @{$ref}{qw/Prefix NamespaceURI/}; | ||||||
321 | # $context->declare_prefix( $prefix, $ns_uri ); | ||||||
322 | $map{$prefix} = $ns_uri; | ||||||
323 | } | ||||||
324 | $self->on_start_prefix_mapping(%map); | ||||||
325 | } | ||||||
326 | |||||||
327 | =head2 on_start_element $elem | ||||||
328 | |||||||
329 | Method handle C |
||||||
330 | |||||||
331 | Method must return C<$elem> or ref to array of objects. | ||||||
332 | |||||||
333 | For example: | ||||||
334 | |||||||
335 | sub on_start_element { | ||||||
336 | my $self = shift; | ||||||
337 | my $elem = shift; | ||||||
338 | $elem->add_content( $self->mk_cdata("test")); | ||||||
339 | return $elem | ||||||
340 | } | ||||||
341 | ... | ||||||
342 | |||||||
343 | return [ $elem, ,$self->mk_element("after_start_elem") ] | ||||||
344 | |||||||
345 | return [ $self->mk_element("before_start_elem"), $elem ] | ||||||
346 | ... | ||||||
347 | |||||||
348 | =cut | ||||||
349 | |||||||
350 | sub on_start_element { | ||||||
351 | shift; | ||||||
352 | return [@_]; | ||||||
353 | } | ||||||
354 | |||||||
355 | sub start_element { | ||||||
356 | my $self = shift; | ||||||
357 | my $data = shift; | ||||||
358 | |||||||
359 | #check current element for skip_content | ||||||
360 | if ( my $current_element = $self->current_element ) { | ||||||
361 | my $skip_content = $current_element->is_skip_content; | ||||||
362 | if ($skip_content) { | ||||||
363 | $current_element->is_skip_content( ++$skip_content ); | ||||||
364 | return; | ||||||
365 | } | ||||||
366 | } | ||||||
367 | my $current_obj = | ||||||
368 | UNIVERSAL::isa( $data, 'XML::Handler::ExtOn::Element' ) | ||||||
369 | ? $data | ||||||
370 | : $self->__mk_element_from_sax2($data); | ||||||
371 | my $res = $self->on_start_element($current_obj); | ||||||
372 | my @stack = $res | ||||||
373 | ? ref($res) eq 'ARRAY' ? @{$res} : ($res) | ||||||
374 | : (); | ||||||
375 | push @stack, $current_obj; | ||||||
376 | my %uniq = (); | ||||||
377 | |||||||
378 | #process answer | ||||||
379 | foreach my $elem (@stack) { | ||||||
380 | |||||||
381 | #clean dups | ||||||
382 | next if $uniq{$elem}++; | ||||||
383 | unless ( $elem eq $current_obj ) { | ||||||
384 | |||||||
385 | # warn "++".$elem->local_name; | ||||||
386 | $self->_process_comm($elem); | ||||||
387 | } | ||||||
388 | else { | ||||||
389 | |||||||
390 | my $res_data = $self->__exp_element_to_sax2($current_obj); | ||||||
391 | |||||||
392 | #register new namespaces | ||||||
393 | my $changes = $current_obj->ns->get_changes; | ||||||
394 | my $parent_map = $current_obj->ns->parent->get_map; | ||||||
395 | |||||||
396 | #warn Dumper( { changes => $changes } ); | ||||||
397 | for ( keys %$changes ) { | ||||||
398 | |||||||
399 | # $self->SUPER::end_prefix_mapping( | ||||||
400 | $self->end_prefix_mapping( | ||||||
401 | { | ||||||
402 | Prefix => $_, | ||||||
403 | NamespaceURI => $parent_map->{$_}, | ||||||
404 | } | ||||||
405 | ) | ||||||
406 | if exists $parent_map->{$_}; | ||||||
407 | |||||||
408 | # $self->SUPER::start_prefix_mapping( | ||||||
409 | $self->start_prefix_mapping( | ||||||
410 | { | ||||||
411 | Prefix => $_, | ||||||
412 | NamespaceURI => $changes->{$_}, | ||||||
413 | } | ||||||
414 | ); | ||||||
415 | } | ||||||
416 | |||||||
417 | #save element in stack | ||||||
418 | push @{ $self->_objects_stack() }, $current_obj; | ||||||
419 | |||||||
420 | #skip deleted elements from xml stream | ||||||
421 | $self->SUPER::start_element($res_data) | ||||||
422 | unless $current_obj->is_delete_element; | ||||||
423 | unless ( $current_obj->is_skip_content ) { | ||||||
424 | $self->_process_comm($_) for @{ $current_obj->_stack }; | ||||||
425 | $current_obj->_stack( [] ); | ||||||
426 | } | ||||||
427 | } | ||||||
428 | |||||||
429 | } | ||||||
430 | } | ||||||
431 | |||||||
432 | =head2 on_end_element $elem | ||||||
433 | |||||||
434 | Method handle C |
||||||
435 | It call before end if element. | ||||||
436 | |||||||
437 | Method must return C<$elem> or ref to array of objects. | ||||||
438 | |||||||
439 | For example: | ||||||
440 | |||||||
441 | sub on_end_element { | ||||||
442 | my $self = shift; | ||||||
443 | my $elem = shift; | ||||||
444 | if ( $elem->is_delete_element ) { | ||||||
445 | warn $elem->local_name . " deleted"; | ||||||
446 | return [ $elem, $self->mk_element("after_deleted_elem") ] | ||||||
447 | }; | ||||||
448 | return $elem | ||||||
449 | } | ||||||
450 | ... | ||||||
451 | |||||||
452 | return [ $elem, ,$self->mk_element("after_close_tag_of_elem") ] | ||||||
453 | |||||||
454 | return [ $self->mk_element("before_close_tag_of_elem"), $elem ] | ||||||
455 | ... | ||||||
456 | |||||||
457 | =cut | ||||||
458 | |||||||
459 | sub on_end_element { | ||||||
460 | shift; | ||||||
461 | return [@_]; | ||||||
462 | } | ||||||
463 | |||||||
464 | sub end_element { | ||||||
465 | my $self = shift; | ||||||
466 | my $data = shift; | ||||||
467 | |||||||
468 | #check current element for skip_content | ||||||
469 | if ( my $current_element = $self->current_element ) { | ||||||
470 | my $skip_content = $current_element->is_skip_content; | ||||||
471 | if ( $skip_content > 1 ) { | ||||||
472 | $current_element->is_skip_content( --$skip_content ); | ||||||
473 | return; | ||||||
474 | } | ||||||
475 | } | ||||||
476 | |||||||
477 | # warn Dumper($data); | ||||||
478 | #pop element from stack | ||||||
479 | my $current_obj = pop @{ $self->_objects_stack() }; | ||||||
480 | |||||||
481 | #setup default ns | ||||||
482 | $data = $current_obj->to_sax2; | ||||||
483 | delete $data->{Attributes}; | ||||||
484 | $data->{NamespaceURI} = $current_obj->default_uri; | ||||||
485 | |||||||
486 | my $res = $self->on_end_element($current_obj); | ||||||
487 | my @stack = $res | ||||||
488 | ? ref($res) eq 'ARRAY' ? @{$res} : ($res) | ||||||
489 | : (); | ||||||
490 | push @stack, $current_obj; | ||||||
491 | my %uniq = (); | ||||||
492 | |||||||
493 | #process answer | ||||||
494 | foreach my $elem (@stack) { | ||||||
495 | |||||||
496 | #clean dups | ||||||
497 | next if $uniq{$elem}++; | ||||||
498 | unless ( $elem eq $current_obj ) { | ||||||
499 | $self->_process_comm($elem); | ||||||
500 | } | ||||||
501 | else { | ||||||
502 | unless ( $current_obj->is_skip_content ) { | ||||||
503 | $self->_process_comm($_) for @{ $current_obj->_stack }; | ||||||
504 | $current_obj->_stack( [] ); | ||||||
505 | } | ||||||
506 | $self->SUPER::end_element($data) | ||||||
507 | unless $current_obj->is_delete_element; | ||||||
508 | my $changes = $current_obj->ns->get_changes; | ||||||
509 | my $parent_map = $current_obj->ns->parent->get_map; | ||||||
510 | for ( keys %$changes ) { | ||||||
511 | $self->end_prefix_mapping( | ||||||
512 | { | ||||||
513 | Prefix => $_, | ||||||
514 | NamespaceURI => $changes->{$_}, | ||||||
515 | } | ||||||
516 | ); | ||||||
517 | if ( exists( $parent_map->{$_} ) ) { | ||||||
518 | $self->start_prefix_mapping( | ||||||
519 | { | ||||||
520 | Prefix => $_, | ||||||
521 | NamespaceURI => $parent_map->{$_}, | ||||||
522 | } | ||||||
523 | ); | ||||||
524 | } | ||||||
525 | } | ||||||
526 | } | ||||||
527 | } | ||||||
528 | } | ||||||
529 | |||||||
530 | =head2 on_characters( $self->current_element, $data->{Data} ) | ||||||
531 | |||||||
532 | Must return string for write to stream. | ||||||
533 | |||||||
534 | sub on_characters { | ||||||
535 | my ( $self, $elem, $str ) = @_; | ||||||
536 | #lowercase all characters | ||||||
537 | return lc $str; | ||||||
538 | } | ||||||
539 | |||||||
540 | |||||||
541 | =cut | ||||||
542 | |||||||
543 | sub on_characters { | ||||||
544 | my ( $self, $elem, $str ) = @_; | ||||||
545 | return $str; | ||||||
546 | } | ||||||
547 | |||||||
548 | =head2 on_cdata ( $current_element, $data ) | ||||||
549 | |||||||
550 | Must return string for write to stream | ||||||
551 | |||||||
552 | sub on_cdata { | ||||||
553 | my ( $self, $elem, $str ) = @_; | ||||||
554 | return lc $str; | ||||||
555 | } | ||||||
556 | |||||||
557 | =cut | ||||||
558 | |||||||
559 | sub on_cdata { | ||||||
560 | my ( $self, $elem, $str ) = @_; | ||||||
561 | return $str; | ||||||
562 | } | ||||||
563 | |||||||
564 | #set flag for cdata content | ||||||
565 | |||||||
566 | sub start_cdata { | ||||||
567 | my $self = shift; | ||||||
568 | $self->_cdata_mode(1); | ||||||
569 | return; | ||||||
570 | } | ||||||
571 | |||||||
572 | #set flag to end cdata | ||||||
573 | |||||||
574 | sub end_cdata { | ||||||
575 | my $self = shift; | ||||||
576 | if ( my $elem = $self->current_element | ||||||
577 | and defined( my $cdata_buf = ${ $self->_cdata_characters } ) ) | ||||||
578 | { | ||||||
579 | if ( defined( my $data = $self->on_cdata( $elem, $cdata_buf ) ) ) { | ||||||
580 | $self->SUPER::start_cdata; | ||||||
581 | $self->SUPER::characters( { Data => $data } ); | ||||||
582 | $self->SUPER::end_cdata; | ||||||
583 | } | ||||||
584 | } | ||||||
585 | |||||||
586 | #after all clear cd_data_buffer and reset cd_data mode flag | ||||||
587 | my $new_buf; | ||||||
588 | $self->_cdata_characters( \$new_buf ); | ||||||
589 | $self->_cdata_mode(0); | ||||||
590 | return; | ||||||
591 | } | ||||||
592 | |||||||
593 | sub characters { | ||||||
594 | my $self = shift; | ||||||
595 | my ($data) = @_; | ||||||
596 | #skip childs elements characters ( > 1 ) and self text ( > 0) | ||||||
597 | # warn $self.Dumper([ map {[caller($_)]} (1..10)]) unless $self->current_element; | ||||||
598 | if ( $self->current_element ) { | ||||||
599 | return if $self->current_element->is_skip_content; | ||||||
600 | } | ||||||
601 | else { | ||||||
602 | |||||||
603 | #skip characters without element | ||||||
604 | return | ||||||
605 | |||||||
606 | # #warn "characters without element" | ||||||
607 | } | ||||||
608 | |||||||
609 | #for cdata section collect characters in buffer | ||||||
610 | if ( $self->_cdata_mode ) { | ||||||
611 | ${ $self->_cdata_characters } .= $data->{Data}; | ||||||
612 | return; | ||||||
613 | } | ||||||
614 | |||||||
615 | #collect chars fo current element | ||||||
616 | if ( | ||||||
617 | defined( | ||||||
618 | my $str = | ||||||
619 | $self->on_characters( $self->current_element, $data->{Data} ) | ||||||
620 | ) | ||||||
621 | ) | ||||||
622 | { | ||||||
623 | return $self->SUPER::characters( { Data => $str } ); | ||||||
624 | } | ||||||
625 | } | ||||||
626 | |||||||
627 | =head2 mk_element |
||||||
628 | |||||||
629 | Return object of element item for include to stream. | ||||||
630 | |||||||
631 | =cut | ||||||
632 | |||||||
633 | sub mk_element { | ||||||
634 | my $self = shift; | ||||||
635 | my $name = shift; | ||||||
636 | my %args = @_; | ||||||
637 | if ( my $current_element = $self->current_element ) { | ||||||
638 | $args{context} = $current_element->ns->sub_context(); | ||||||
639 | } | ||||||
640 | $args{context} ||= $self->context->sub_context(); | ||||||
641 | my $elem = new XML::Handler::ExtOn::Element:: | ||||||
642 | name => $name, | ||||||
643 | %args; | ||||||
644 | return $elem; | ||||||
645 | } | ||||||
646 | |||||||
647 | =head2 mk_from_xml |
||||||
648 | |||||||
649 | Return command for include to stream. | ||||||
650 | |||||||
651 | =cut | ||||||
652 | |||||||
653 | sub mk_from_xml { | ||||||
654 | my $self = shift; | ||||||
655 | my $string = shift; | ||||||
656 | my $skip_tmp_root = XML::Handler::ExtOn::IncXML->new( Handler => $self ); | ||||||
657 | my $sax2_filter = XML::Filter::SAX1toSAX2->new( Handler => $skip_tmp_root ); | ||||||
658 | my $parser = XML::Parser::PerlSAX->new( | ||||||
659 | { | ||||||
660 | Handler => $sax2_filter, | ||||||
661 | Source => { String => " |
||||||
662 | } | ||||||
663 | ); | ||||||
664 | return $parser; | ||||||
665 | } | ||||||
666 | |||||||
667 | =head2 mk_cdata $string | \$string | ||||||
668 | |||||||
669 | return command for insert cdata to stream | ||||||
670 | |||||||
671 | =cut | ||||||
672 | |||||||
673 | sub mk_cdata { | ||||||
674 | my $self = shift; | ||||||
675 | my $string = shift; | ||||||
676 | return { type => 'CDATA', data => ref($string) ? $string : \$string }; | ||||||
677 | } | ||||||
678 | |||||||
679 | =head2 mk_characters $string | \$string | ||||||
680 | |||||||
681 | return command for insert characters to stream | ||||||
682 | |||||||
683 | =cut | ||||||
684 | |||||||
685 | sub mk_characters { | ||||||
686 | my $self = shift; | ||||||
687 | my $string = shift; | ||||||
688 | return { type => 'CHARACTERS', data => ref($string) ? $string : \$string }; | ||||||
689 | } | ||||||
690 | |||||||
691 | sub __mk_element_from_sax2 { | ||||||
692 | my $self = shift; | ||||||
693 | my $data = shift; | ||||||
694 | my $elem = $self->mk_element( $data->{LocalName}, sax2 => $data, @_ ); | ||||||
695 | return $elem; | ||||||
696 | } | ||||||
697 | |||||||
698 | sub __exp_element_to_sax2 { | ||||||
699 | my $self = shift; | ||||||
700 | my $elem = shift; | ||||||
701 | return $elem->to_sax2; | ||||||
702 | } | ||||||
703 | |||||||
704 | =head2 current_element | ||||||
705 | |||||||
706 | Return link to current processing element. | ||||||
707 | |||||||
708 | =cut | ||||||
709 | |||||||
710 | sub current_element { | ||||||
711 | my $self = shift; | ||||||
712 | if ( my $stack = $self->_objects_stack() ) { | ||||||
713 | return $stack->[-1]; | ||||||
714 | } | ||||||
715 | return; | ||||||
716 | } | ||||||
717 | |||||||
718 | # Private method for process commands | ||||||
719 | |||||||
720 | sub _process_comm { | ||||||
721 | my $self = shift; | ||||||
722 | my $comm = shift || return; | ||||||
723 | if ( UNIVERSAL::isa( $comm, 'XML::Parser::PerlSAX' ) ) { | ||||||
724 | $comm->parse; | ||||||
725 | } | ||||||
726 | elsif ( UNIVERSAL::isa( $comm, 'XML::Handler::ExtOn::Element' ) ) { | ||||||
727 | $self->start_element($comm); | ||||||
728 | |||||||
729 | while ( my $obj = shift @{ $comm->_stack } ) { | ||||||
730 | $self->_process_comm($obj); | ||||||
731 | } | ||||||
732 | $self->end_element($comm); | ||||||
733 | } | ||||||
734 | elsif ( ref($comm) eq 'HASH' and exists $comm->{type} ) { | ||||||
735 | if ( $comm->{type} eq 'CDATA' ) { | ||||||
736 | $self->start_cdata; | ||||||
737 | $self->characters( { Data => ${ $comm->{data} } } ); | ||||||
738 | $self->end_cdata; | ||||||
739 | } | ||||||
740 | elsif ( $comm->{type} eq 'CHARACTERS' ) { | ||||||
741 | $self->characters( { Data => ${ $comm->{data} } } ); | ||||||
742 | } | ||||||
743 | } | ||||||
744 | else { | ||||||
745 | warn " Unknown DATA $comm"; | ||||||
746 | } | ||||||
747 | } | ||||||
748 | |||||||
749 | =head2 add_namespace |
||||||
750 | |||||||
751 | Add Namespace mapping. return C<$self> | ||||||
752 | |||||||
753 | If C |
||||||
754 | that have no prefix. | ||||||
755 | |||||||
756 | $elem->add_namespace( | ||||||
757 | "myns" => 'http://example.com/myns', | ||||||
758 | "myns_test", 'http://example.com/myns_test', | ||||||
759 | ''=>'http://example.com/new_default_namespace' | ||||||
760 | ); | ||||||
761 | |||||||
762 | =cut | ||||||
763 | |||||||
764 | sub add_namespace { | ||||||
765 | my $self = shift; | ||||||
766 | my $context = $self->context; | ||||||
767 | if ( my $current = $self->current_element ) { | ||||||
768 | $context = $current->ns; | ||||||
769 | } | ||||||
770 | my %map = @_; | ||||||
771 | while ( my ($prefix, $ns_uri ) = each %map ) { | ||||||
772 | $context->declare_prefix( $prefix, $ns_uri ); | ||||||
773 | } | ||||||
774 | } | ||||||
775 | |||||||
776 | 1; | ||||||
777 | __END__ |