File Coverage

blib/lib/HTML/Element/Library.pm
Criterion Covered Total %
statement 25 27 92.5
branch n/a
condition n/a
subroutine 9 9 100.0
pod n/a
total 34 36 94.4


line stmt bran cond sub pod time code
1             package HTML::Element::Library;
2 2     2   79968 use strict;
  2         4  
  2         88  
3 2     2   12 use warnings;
  2         3  
  2         122  
4              
5             our $VERSION = '5.210000';
6             our $DEBUG = 0;
7              
8 2     2   1062 use Array::Group ':all';
  2         1277  
  2         332  
9 2     2   12 use Carp 'confess';
  2         4  
  2         92  
10 2     2   1335 use Data::Dumper;
  2         18905  
  2         182  
11 2     2   1124 use Data::Rmap 'rmap_array';
  2         2199  
  2         134  
12 2     2   10 use HTML::Element;
  2         4  
  2         12  
13 2     2   1057 use HTML::FillInForm;
  2         4810  
  2         86  
14 2     2   156207 use List::MoreUtils ':all';
  0            
  0            
15             use List::Rotation::Cycle;
16             use List::Util 'first';
17             use Params::Validate ':all';
18             use Scalar::Listify;
19              
20             # https://rt.cpan.org/Ticket/Display.html?id=44105
21             sub HTML::Element::fillinform {
22             my ($tree, $hashref, $return_tree, $guts) = @_;
23             (ref $hashref) eq 'HASH' or confess 'hashref not supplied as argument' ;
24              
25             my $html = $tree->as_HTML;
26             my $new_html = HTML::FillInForm->fill(\$html, $hashref);
27              
28             if ($return_tree) {
29             $tree = HTML::TreeBuilder->new_from_content($new_html);
30             $tree = $guts ? $tree->guts : $tree ;
31             } else {
32             $new_html;
33             }
34             }
35              
36             sub HTML::Element::siblings {
37             my $element = shift;
38             my $p = $element->parent;
39             return () unless $p;
40             $p->content_list;
41             }
42              
43             sub HTML::Element::defmap {
44             my($tree, $attr, $hashref, $debug) = @_;
45              
46             while (my ($k, $v) = (each %$hashref)) {
47             warn "defmap looks for ($attr => $k)" if $debug;
48             my $found = $tree->look_down($attr => $k);
49             if ($found) {
50             warn "($attr => $k) was found.. replacing with '$v'" if $debug;
51             $found->replace_content( $v );
52             }
53             }
54             }
55              
56             sub HTML::Element::_only_empty_content {
57             my ($self) = @_;
58             my @c = $self->content_list;
59             my $length = scalar @c;
60              
61             scalar @c == 1 and not length $c[0];
62             }
63              
64             sub HTML::Element::prune {
65             my ($self) = @_;
66              
67             for my $c ($self->content_list) {
68             next unless ref $c;
69             $c->prune;
70             }
71              
72             # post-order:
73             $self->delete if ($self->is_empty or $self->_only_empty_content);
74             $self;
75             }
76              
77             sub HTML::Element::newchild {
78             my ($lol, $parent_label, @newchild) = @_;
79             rmap_array {
80             if ($_->[0] eq $parent_label) {
81             $_ = [ $parent_label => @newchild ];
82             Data::Rmap::cut($_);
83             } else {
84             $_;
85             }
86             } $lol;
87             }
88              
89             sub HTML::Element::crunch { ## no critic (RequireArgUnpacking)
90             my $container = shift;
91              
92             my %p = validate(@_, {
93             look_down => { type => ARRAYREF },
94             leave => { default => 1 },
95             });
96              
97             my @look_down = @{$p{look_down}} ;
98             my @elem = $container->look_down(@look_down) ;
99              
100             my $detached;
101              
102             for my $elem (@elem) {
103             $elem->detach if $detached++ >= $p{leave};
104             }
105             }
106              
107             sub HTML::Element::hash_map { ## no critic (RequireArgUnpacking)
108             my $container = shift;
109              
110             my %p = validate(@_, {
111             hash => { type => HASHREF },
112             to_attr => 1,
113             excluding => { type => ARRAYREF , default => [] },
114             debug => { default => 0 },
115             });
116              
117             warn 'The container tag is ', $container->tag if $p{debug} ;
118             warn 'hash' . Dumper($p{hash}) if $p{debug} ;
119             #warn 'at_under' . Dumper(\@_) if $p{debug} ;
120              
121             my @same_as = $container->look_down( $p{to_attr} => qr/.+/s ) ;
122              
123             warn 'Found ' . scalar(@same_as) . ' nodes' if $p{debug} ;
124              
125             for my $same_as (@same_as) {
126             my $attr_val = $same_as->attr($p{to_attr}) ;
127             if (first { $attr_val eq $_ } @{$p{excluding}}) {
128             warn "excluding $attr_val" if $p{debug} ;
129             next;
130             }
131             warn "processing $attr_val" if $p{debug} ;
132             $same_as->replace_content($p{hash}->{$attr_val});
133             }
134             }
135              
136             sub HTML::Element::hashmap {
137             my ($container, $attr_name, $hashref, $excluding, $debug) = @_;
138              
139             $excluding ||= [] ;
140              
141             $container->hash_map(
142             hash => $hashref,
143             to_attr => $attr_name,
144             excluding => $excluding,
145             debug => $debug);
146             }
147              
148              
149             sub HTML::Element::passover {
150             my ($tree, @to_preserve) = @_;
151              
152             warn "ARGS: my ($tree, @to_preserve)" if $DEBUG;
153             warn $tree->as_HTML(undef, ' ') if $DEBUG;
154              
155             my $exodus = $tree->look_down(id => $to_preserve[0]);
156              
157             warn "E: $exodus" if $DEBUG;
158              
159             my @s = HTML::Element::siblings($exodus);
160              
161             for my $s (@s) {
162             next unless ref $s;
163             $s->delete unless first { $s->attr('id') eq $_ } @to_preserve;
164             }
165              
166             return $exodus; # Goodbye Egypt! http://en.wikipedia.org/wiki/Passover
167             }
168              
169             sub HTML::Element::sibdex {
170             my $element = shift;
171             firstidx { $_ eq $element } $element->siblings
172             }
173              
174             sub HTML::Element::addr { goto &HTML::Element::sibdex }
175              
176             sub HTML::Element::replace_content {
177             my $elem = shift;
178             $elem->delete_content;
179             $elem->push_content(@_);
180             }
181              
182             sub HTML::Element::wrap_content {
183             my($self, $wrap) = @_;
184             my $content = $self->content;
185             if (ref $content) {
186             $wrap->push_content(@$content);
187             @$content = ($wrap);
188             }
189             else {
190             $self->push_content($wrap);
191             }
192             $wrap;
193             }
194              
195             sub HTML::Element::Library::super_literal {
196             my($text) = @_;
197             HTML::Element->new('~literal', text => $text);
198             }
199              
200             sub HTML::Element::position {
201             # Report coordinates by chasing addr's up the
202             # HTML::ElementSuper tree. We know we've reached
203             # the top when a) there is no parent, or b) the
204             # parent is some HTML::Element unable to report
205             # it's position.
206             my $p = shift;
207             my @pos;
208             while ($p) {
209             my $a = $p->addr;
210             unshift @pos, $a if defined $a;
211             $p = $p->parent;
212             }
213             @pos;
214             }
215              
216             sub HTML::Element::content_handler {
217             my ($tree, %content_hash) = @_;
218              
219             for my $k (keys %content_hash) {
220             $tree->set_child_content(id => $k, $content_hash{$k});
221             }
222             }
223              
224             sub HTML::Element::assign { goto &HTML::Element::content_handler }
225              
226             sub make_counter {
227             my $i = 1;
228             sub {
229             shift() . ':' . $i++
230             }
231             }
232              
233             sub HTML::Element::iter {
234             my ($tree, $p, @data) = @_;
235              
236             # warn 'P: ' , $p->attr('id') ;
237             # warn 'H: ' , $p->as_HTML;
238              
239             # my $id_incr = make_counter;
240             my @item = map {
241             my $new_item = clone $p;
242             $new_item->replace_content($_);
243             $new_item;
244             } @data;
245              
246             $p->replace_with(@item);
247             }
248              
249             sub HTML::Element::iter2 { ## no critic (RequireArgUnpacking)
250             my $tree = shift;
251              
252             #warn "INPUT TO TABLE2: ", Dumper \@_;
253              
254             my %p = validate(
255             @_, {
256             wrapper_ld => { default => ['_tag' => 'dl'] },
257             wrapper_data => 1,
258             wrapper_proc => { default => undef },
259             item_ld => {
260             default => sub {
261             my $tr = shift;
262             [
263             $tr->look_down('_tag' => 'dt'),
264             $tr->look_down('_tag' => 'dd')
265             ];
266             }},
267             item_data => {
268             default => sub {
269             my ($wrapper_data) = @_;
270             shift @{$wrapper_data};
271             }},
272             item_proc => {
273             default => sub {
274             my ($item_elems, $item_data, $row_count) = @_;
275             $item_elems->[$_]->replace_content($item_data->[$_]) for (0,1) ;
276             $item_elems;
277             }},
278             splice => {
279             default => sub {
280             my ($container, @item_elems) = @_;
281             $container->splice_content(0, 2, @item_elems);
282             }
283             },
284             debug => {default => 0}
285             }
286             );
287              
288             warn 'wrapper_data: ' . Dumper $p{wrapper_data} if $p{debug} ;
289              
290             my $container = ref_or_ld($tree, $p{wrapper_ld});
291             warn 'container: ' . $container if $p{debug} ;
292             warn 'wrapper_(preproc): ' . $container->as_HTML if $p{debug} ;
293             $p{wrapper_proc}->($container) if defined $p{wrapper_proc} ;
294             warn 'wrapper_(postproc): ' . $container->as_HTML if $p{debug} ;
295              
296             my $_item_elems = $p{item_ld}->($container);
297              
298             my $row_count;
299             my @item_elem;
300             while(1){
301             my $item_data = $p{item_data}->($p{wrapper_data});
302             last unless defined $item_data;
303              
304             warn Dumper('item_data', $item_data) if $p{debug};
305              
306             my $item_elems = [ map { $_->clone } @{$_item_elems} ] ;
307              
308             if ($p{debug}) {
309             for (@{$item_elems}) {
310             warn 'ITEM_ELEMS ', $_->as_HTML if $p{debug};
311             }
312             }
313              
314             my $new_item_elems = $p{item_proc}->($item_elems, $item_data, ++$row_count);
315              
316             if ($p{debug}) {
317             for (@{$new_item_elems}) {
318             warn 'NEWITEM_ELEMS ', $_->as_HTML if $p{debug};
319             }
320             }
321              
322             push @item_elem, @{$new_item_elems} ;
323             }
324              
325             warn 'pushing ' . @item_elem . ' elems' if $p{debug} ;
326              
327             $p{splice}->($container, @item_elem);
328             }
329              
330             sub HTML::Element::dual_iter {
331             my ($parent, $data) = @_;
332              
333             my ($prototype_a, $prototype_b) = $parent->content_list;
334              
335             # my $id_incr = make_counter;
336              
337             my $i;
338              
339             @$data %2 == 0 or confess 'dataset does not contain an even number of members';
340              
341             my @iterable_data = ngroup 2 => @$data;
342              
343             my @item = map {
344             my ($new_a, $new_b) = map { clone $_ } ($prototype_a, $prototype_b) ;
345             $new_a->splice_content(0,1, $_->[0]);
346             $new_b->splice_content(0,1, $_->[1]);
347             #$_->attr('id', $id_incr->($_->attr('id'))) for ($new_a, $new_b) ;
348             ($new_a, $new_b)
349             } @iterable_data;
350              
351             $parent->splice_content(0, 2, @item);
352             }
353              
354             sub HTML::Element::set_child_content { ## no critic (RequireArgUnpacking)
355             my $tree = shift;
356             my $content = pop;
357             my @look_down = @_;
358              
359             my $content_tag = $tree->look_down(@look_down);
360              
361             unless ($content_tag) {
362             warn "criteria [@look_down] not found";
363             return;
364             }
365              
366             $content_tag->replace_content($content);
367             }
368              
369             sub HTML::Element::highlander {
370             my ($tree, $local_root_id, $aref, @arg) = @_;
371              
372             ref $aref eq 'ARRAY' or confess 'must supply array reference';
373              
374             my @aref = @$aref;
375             @aref % 2 == 0 or confess 'supplied array ref must have an even number of entries';
376              
377             warn __PACKAGE__ if $DEBUG;
378              
379             my $survivor;
380             while (my ($id, $test) = splice @aref, 0, 2) {
381             warn $id if $DEBUG;
382             if ($test->(@arg)) {
383             $survivor = $id;
384             last;
385             }
386             }
387              
388             my @id_survivor = (id => $survivor);
389             my $survivor_node = $tree->look_down(@id_survivor);
390             # warn $survivor;
391             # warn $local_root_id;
392             # warn $node;
393              
394             warn "survivor: $survivor" if $DEBUG;
395             warn 'tree: ' . $tree->as_HTML if $DEBUG;
396              
397             $survivor_node or die "search for @id_survivor failed in tree($tree): " . $tree->as_HTML;
398              
399             my $survivor_node_parent = $survivor_node->parent;
400             $survivor_node = $survivor_node->clone;
401             $survivor_node_parent->replace_content($survivor_node);
402              
403             warn 'new tree: ' . $tree->as_HTML if $DEBUG;
404              
405             $survivor_node;
406             }
407              
408             sub HTML::Element::highlander2 { ## no critic (RequireArgUnpacking)
409             my $tree = shift;
410              
411             my %p = validate(@_, {
412             cond => { type => ARRAYREF },
413             cond_arg => {
414             type => ARRAYREF,
415             default => []
416             },
417             debug => { default => 0 }
418             });
419              
420             my @cond = @{$p{cond}};
421             @cond % 2 == 0 or confess 'supplied array ref must have an even number of entries';
422              
423             warn __PACKAGE__ if $p{debug};
424              
425             my @cond_arg = @{$p{cond_arg}};
426              
427             my $survivor; my $then;
428             while (my ($id, $if_then) = splice @cond, 0, 2) {
429             warn $id if $p{debug};
430             my ($if, $_then);
431              
432             if (ref $if_then eq 'ARRAY') {
433             ($if, $_then) = @$if_then;
434             } else {
435             ($if, $_then) = ($if_then, sub {});
436             }
437              
438             if ($if->(@cond_arg)) {
439             $survivor = $id;
440             $then = $_then;
441             last;
442             }
443             }
444              
445             my @ld = (ref $survivor eq 'ARRAY') ? @$survivor : (id => $survivor);
446              
447             warn 'survivor: ', $survivor if $p{debug};
448             warn 'survivor_ld: ', Dumper \@ld if $p{debug};
449              
450             my $survivor_node = $tree->look_down(@ld);
451              
452             $survivor_node or confess "search for @ld failed in tree($tree): " . $tree->as_HTML;
453              
454             my $survivor_node_parent = $survivor_node->parent;
455             $survivor_node = $survivor_node->clone;
456             $survivor_node_parent->replace_content($survivor_node);
457              
458             # **************** NEW FUNCTIONALITY *******************
459             # apply transforms on survivor node
460              
461             warn 'SURV::pre_trans ' . $survivor_node->as_HTML if $p{debug};
462             $then->($survivor_node, @cond_arg);
463             warn 'SURV::post_trans ' . $survivor_node->as_HTML if $p{debug};
464             # **************** NEW FUNCTIONALITY *******************
465              
466             $survivor_node;
467             }
468              
469             sub overwrite_action {
470             my ($mute_node, %X) = @_;
471              
472             $mute_node->attr($X{local_attr}{name} => $X{local_attr}{value}{new});
473             }
474              
475             sub HTML::Element::overwrite_attr {
476             my $tree = shift;
477              
478             $tree->mute_elem(@_, \&overwrite_action);
479             }
480              
481             sub HTML::Element::mute_elem {
482             my ($tree, $mute_attr, $closures, $post_hook) = @_;
483              
484             my @mute_node = $tree->look_down($mute_attr => qr/.*/s) ;
485              
486             for my $mute_node (@mute_node) {
487             my ($local_attr,$mute_key) = split /\s+/s, $mute_node->attr($mute_attr);
488             my $local_attr_value_current = $mute_node->attr($local_attr);
489             my $local_attr_value_new = $closures->{$mute_key}->($tree, $mute_node, $local_attr_value_current);
490             $post_hook->(
491             $mute_node,
492             tree => $tree,
493             local_attr => {
494             name => $local_attr,
495             value => {
496             current => $local_attr_value_current,
497             new => $local_attr_value_new
498             }
499             }
500             ) if ($post_hook) ;
501             }
502             }
503              
504              
505              
506             sub HTML::Element::table {
507             my ($s, %table) = @_;
508             my $table = {};
509              
510             # Get the table element
511             $table->{table_node} = $s->look_down(id => $table{gi_table});
512             $table->{table_node} or confess "table tag not found via (id => $table{gi_table}";
513              
514             # Get the prototype tr element(s)
515             my @table_gi_tr = listify $table{gi_tr} ;
516             my @iter_node = map {
517             my $tr = $table->{table_node}->look_down(id => $_);
518             $tr or confess "tr with id => $_ not found";
519             $tr;
520             } @table_gi_tr;
521              
522             warn 'found ' . @iter_node . ' iter nodes ' if $DEBUG;
523             my $iter_node = List::Rotation::Cycle->new(@iter_node);
524              
525             # warn $iter_node;
526             warn Dumper ($iter_node, \@iter_node) if $DEBUG;
527              
528             # $table->{content} = $table{content};
529             # $table->{parent} = $table->{table_node}->parent;
530              
531             # $table->{table_node}->detach;
532             # $_->detach for @iter_node;
533              
534             my @table_rows;
535              
536             while (1) {
537             my $row = $table{tr_data}->($table, $table{table_data});
538             last unless defined $row;
539              
540             # get a sample table row and clone it.
541             my $I = $iter_node->next;
542             warn "I: $I" if $DEBUG;
543             my $new_iter_node = $I->clone;
544              
545             $table{td_data}->($new_iter_node, $row);
546             push @table_rows, $new_iter_node;
547             }
548              
549             if (@table_rows) {
550             my $replace_with_elem = $s->look_down(id => shift @table_gi_tr) ;
551             $s->look_down(id => $_)->detach for @table_gi_tr;
552             $replace_with_elem->replace_with(@table_rows);
553             }
554             }
555              
556             sub ref_or_ld {
557             my ($tree, $slot) = @_;
558              
559             if (ref($slot) eq 'CODE') {
560             $slot->($tree);
561             } else {
562             $tree->look_down(@$slot);
563             }
564             }
565              
566             sub HTML::Element::table2 { ## no critic (RequireArgUnpacking)
567             my $tree = shift;
568              
569             my %p = validate(
570             @_, {
571             table_ld => { default => ['_tag' => 'table'] },
572             table_data => 1,
573             table_proc => { default => undef },
574             tr_ld => { default => ['_tag' => 'tr'] },
575             tr_data => {
576             default => sub {
577             my ($self, $data) = @_;
578             shift @{$data};
579             }},
580             tr_base_id => { default => undef },
581             tr_proc => { default => sub {} },
582             td_proc => 1,
583             debug => {default => 0}
584             }
585             );
586              
587             warn 'INPUT TO TABLE2: ', Dumper \@_ if $p{debug};
588             warn 'table_data: ' . Dumper $p{table_data} if $p{debug} ;
589              
590             my $table = {};
591              
592             # Get the table element
593             $table->{table_node} = ref_or_ld( $tree, $p{table_ld} ) ;
594             $table->{table_node} or confess 'table tag not found via ' . Dumper($p{table_ld}) ;
595              
596             warn 'table: ' . $table->{table_node}->as_HTML if $p{debug};
597              
598             # Get the prototype tr element(s)
599             my @proto_tr = ref_or_ld( $table->{table_node}, $p{tr_ld} ) ;
600              
601             warn 'found ' . @proto_tr . ' iter nodes' if $p{debug};
602              
603             return unless @proto_tr;
604              
605             if ($p{debug}) {
606             warn $_->as_HTML for @proto_tr;
607             }
608             my $proto_tr = List::Rotation::Cycle->new(@proto_tr);
609              
610             my $tr_parent = $proto_tr[0]->parent;
611             warn 'parent element of trs: ' . $tr_parent->as_HTML if $p{debug};
612              
613             my $row_count;
614              
615             my @table_rows;
616              
617             while(1) {
618             my $row = $p{tr_data}->($table, $p{table_data}, $row_count);
619             warn 'data row: ' . Dumper $row if $p{debug};
620             last unless defined $row;
621              
622             # wont work: my $new_iter_node = $table->{iter_node}->clone;
623             my $new_tr_node = $proto_tr->next->clone;
624             warn "new_tr_node: $new_tr_node" if $p{debug};
625              
626             $p{tr_proc}->($tree, $new_tr_node, $row, $p{tr_base_id}, ++$row_count) if defined $p{tr_proc};
627              
628             warn 'data row redux: ' . Dumper $row if $p{debug};
629              
630             $p{td_proc}->($new_tr_node, $row);
631             push @table_rows, $new_tr_node;
632             }
633              
634             $_->detach for @proto_tr;
635              
636             $tr_parent->push_content(@table_rows) if (@table_rows) ;
637             }
638              
639             sub HTML::Element::unroll_select {
640             my ($s, %select) = @_;
641              
642             my $select = {};
643             warn 'Select Hash: ' . Dumper(\%select) if $select{debug};
644              
645             my $select_node = $s->look_down(id => $select{select_label});
646             warn "Select Node: $select_node" if $select{debug};
647              
648             unless ($select{append}) {
649             for my $option ($select_node->look_down('_tag' => 'option')) {
650             $option->delete;
651             }
652             }
653              
654             my $option = HTML::Element->new('option');
655             warn "Option Node: $option" if $select{debug};
656              
657             $option->detach;
658              
659             while (my $row = $select{data_iter}->($select{data})) {
660             warn 'Data Row: ' . Dumper($row) if $select{debug};
661             my $o = $option->clone;
662             $o->attr('value', $select{option_value}->($row));
663             $o->attr('SELECTED', 1) if (exists $select{option_selected} and $select{option_selected}->($row));
664              
665             $o->replace_content($select{option_content}->($row));
666             $select_node->push_content($o);
667             warn $o->as_HTML if $select{debug};
668             }
669             }
670              
671             sub HTML::Element::set_sibling_content {
672             my ($elt, $content) = @_;
673              
674             $elt->parent->splice_content($elt->pindex + 1, 1, $content);
675             }
676              
677             sub HTML::TreeBuilder::parse_string {
678             my ($package, $string) = @_;
679              
680             my $h = HTML::TreeBuilder->new;
681             HTML::TreeBuilder->parse($string);
682             }
683              
684             1;
685             __END__