This is a live mirror of the Perl 5 development currently hosted at https://github.com/perl/perl5
L<PerlIO> and Pod::Html
[perl5.git] / lib / Pod / Simple.pm
CommitLineData
351625bd
SP
1
2require 5;
3package Pod::Simple;
4use strict;
5use Carp ();
6BEGIN { *DEBUG = sub () {0} unless defined &DEBUG }
7use integer;
8use Pod::Escapes 1.03 ();
9use Pod::Simple::LinkSection ();
10use Pod::Simple::BlackBox ();
11#use utf8;
12
13use vars qw(
14 $VERSION @ISA
15 @Known_formatting_codes @Known_directives
16 %Known_formatting_codes %Known_directives
17 $NL
18);
19
20@ISA = ('Pod::Simple::BlackBox');
9ea6f39e 21$VERSION = '3.04';
351625bd
SP
22
23@Known_formatting_codes = qw(I B C L E F S X Z);
24%Known_formatting_codes = map(($_=>1), @Known_formatting_codes);
25@Known_directives = qw(head1 head2 head3 head4 item over back);
26%Known_directives = map(($_=>'Plain'), @Known_directives);
27$NL = $/ unless defined $NL;
28
29#-----------------------------------------------------------------------------
30# Set up some constants:
31
32BEGIN {
33 if(defined &ASCII) { }
34 elsif(chr(65) eq 'A') { *ASCII = sub () {1} }
35 else { *ASCII = sub () {''} }
36
37 unless(defined &MANY_LINES) { *MANY_LINES = sub () {20} }
38 DEBUG > 4 and print "MANY_LINES is ", MANY_LINES(), "\n";
39 unless(MANY_LINES() >= 1) {
40 die "MANY_LINES is too small (", MANY_LINES(), ")!\nAborting";
41 }
42 if(defined &UNICODE) { }
43 elsif($] >= 5.008) { *UNICODE = sub() {1} }
44 else { *UNICODE = sub() {''} }
45}
46if(DEBUG > 2) {
47 print "# We are ", ASCII ? '' : 'not ', "in ASCII-land\n";
48 print "# We are under a Unicode-safe Perl.\n";
49}
50
51# Design note:
52# This is a parser for Pod. It is not a parser for the set of Pod-like
53# languages which happens to contain Pod -- it is just for Pod, plus possibly
54# some extensions.
55
56# @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @
57#@ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @ @
58#@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
59
60__PACKAGE__->_accessorize(
61 'nbsp_for_S', # Whether to map S<...>'s to \xA0 characters
62 'source_filename', # Filename of the source, for use in warnings
63 'source_dead', # Whether to consider this parser's source dead
64
65 'output_fh', # The filehandle we're writing to, if applicable.
66 # Used only in some derived classes.
67
68 'hide_line_numbers', # For some dumping subclasses: whether to pointedly
69 # suppress the start_line attribute
70
71 'line_count', # the current line number
72 'pod_para_count', # count of pod paragraphs seen so far
73
74 'no_whining', # whether to suppress whining
75 'no_errata_section', # whether to suppress the errata section
76 'complain_stderr', # whether to complain to stderr
77
78 'doc_has_started', # whether we've fired the open-Document event yet
79
80 'bare_output', # For some subclasses: whether to prepend
81 # header-code and postpend footer-code
82
83 'fullstop_space_harden', # Whether to turn ". " into ".[nbsp] ";
84
85 'nix_X_codes', # whether to ignore X<...> codes
86 'merge_text', # whether to avoid breaking a single piece of
87 # text up into several events
88
89 'preserve_whitespace', # whether to try to keep whitespace as-is
90
91 'content_seen', # whether we've seen any real Pod content
92 'errors_seen', # TODO: document. whether we've seen any errors (fatal or not)
93
94 'codes_in_verbatim', # for PseudoPod extensions
95
96 'code_handler', # coderef to call when a code (non-pod) line is seen
97 'cut_handler', # coderef to call when a =cut line is seen
98 #Called like:
99 # $code_handler->($line, $self->{'line_count'}, $self) if $code_handler;
100 # $cut_handler->($line, $self->{'line_count'}, $self) if $cut_handler;
101
102);
103
104#@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
105
106sub any_errata_seen { # good for using as an exit() value...
107 return shift->{'errors_seen'} || 0;
108}
109
110#@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
111# Pull in some functions that, for some reason, I expect to see here too:
112BEGIN {
113 *pretty = \&Pod::Simple::BlackBox::pretty;
114 *stringify_lol = \&Pod::Simple::BlackBox::stringify_lol;
115}
116
117#@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
118
119sub version_report {
120 my $class = ref($_[0]) || $_[0];
121 if($class eq __PACKAGE__) {
122 return "$class $VERSION";
123 } else {
124 my $v = $class->VERSION;
125 return "$class $v (" . __PACKAGE__ . " $VERSION)";
126 }
127}
128
129#@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
130
131#sub curr_open { # read-only list accessor
132# return @{ $_[0]{'curr_open'} || return() };
133#}
134#sub _curr_open_listref { $_[0]{'curr_open'} ||= [] }
135
136
137sub output_string {
138 # Works by faking out output_fh. Simplifies our code.
139 #
140 my $this = shift;
141 return $this->{'output_string'} unless @_; # GET.
142
143 require Pod::Simple::TiedOutFH;
144 my $x = (defined($_[0]) and ref($_[0])) ? $_[0] : \( $_[0] );
145 $$x = '' unless defined $$x;
146 DEBUG > 4 and print "# Output string set to $x ($$x)\n";
147 $this->{'output_fh'} = Pod::Simple::TiedOutFH->handle_on($_[0]);
148 return
149 $this->{'output_string'} = $_[0];
150 #${ ${ $this->{'output_fh'} } };
151}
152
153sub abandon_output_string { $_[0]->abandon_output_fh; delete $_[0]{'output_string'} }
154sub abandon_output_fh { $_[0]->output_fh(undef) }
155# These don't delete the string or close the FH -- they just delete our
156# references to it/them.
157# TODO: document these
158
159#@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
160
161sub new {
162 # takes no parameters
163 my $class = ref($_[0]) || $_[0];
164 #Carp::croak(__PACKAGE__ . " is a virtual base class -- see perldoc "
165 # . __PACKAGE__ );
166 return bless {
167 'accept_codes' => { map( ($_=>$_), @Known_formatting_codes ) },
168 'accept_directives' => { %Known_directives },
169 'accept_targets' => {},
170 }, $class;
171}
172
173
174
175# TODO: an option for whether to interpolate E<...>'s, or just resolve to codes.
176
177#@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
178
179sub _handle_element_start { # OVERRIDE IN DERIVED CLASS
180 my($self, $element_name, $attr_hash_r) = @_;
181 return;
182}
183
184sub _handle_element_end { # OVERRIDE IN DERIVED CLASS
185 my($self, $element_name) = @_;
186 return;
187}
188
189sub _handle_text { # OVERRIDE IN DERIVED CLASS
190 my($self, $text) = @_;
191 return;
192}
193
194#@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
195#
196# And now directives (not targets)
197
198sub accept_directive_as_verbatim { shift->_accept_directives('Verbatim', @_) }
199sub accept_directive_as_data { shift->_accept_directives('Data', @_) }
200sub accept_directive_as_processed { shift->_accept_directives('Plain', @_) }
201
202sub _accept_directives {
203 my($this, $type) = splice @_,0,2;
204 foreach my $d (@_) {
205 next unless defined $d and length $d;
206 Carp::croak "\"$d\" isn't a valid directive name"
207 unless $d =~ m/^[a-zA-Z][a-zA-Z0-9]*$/s;
208 Carp::croak "\"$d\" is already a reserved Pod directive name"
209 if exists $Known_directives{$d};
210 $this->{'accept_directives'}{$d} = $type;
211 DEBUG > 2 and print "Learning to accept \"=$d\" as directive of type $type\n";
212 }
213 DEBUG > 6 and print "$this\'s accept_directives : ",
214 pretty($this->{'accept_directives'}), "\n";
215
216 return sort keys %{ $this->{'accept_directives'} } if wantarray;
217 return;
218}
219
220#--------------------------------------------------------------------------
221# TODO: document these:
222
223sub unaccept_directive { shift->unaccept_directives(@_) };
224
225sub unaccept_directives {
226 my $this = shift;
227 foreach my $d (@_) {
228 next unless defined $d and length $d;
229 Carp::croak "\"$d\" isn't a valid directive name"
230 unless $d =~ m/^[a-zA-Z][a-zA-Z0-9]*$/s;
231 Carp::croak "But you must accept \"$d\" directives -- it's a builtin!"
232 if exists $Known_directives{$d};
233 delete $this->{'accept_directives'}{$d};
234 DEBUG > 2 and print "OK, won't accept \"=$d\" as directive.\n";
235 }
236 return sort keys %{ $this->{'accept_directives'} } if wantarray;
237 return
238}
239
240#@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
241#
242# And now targets (not directives)
243
244sub accept_target { shift->accept_targets(@_) } # alias
245sub accept_target_as_text { shift->accept_targets_as_text(@_) } # alias
246
247
248sub accept_targets { shift->_accept_targets('1', @_) }
249
250sub accept_targets_as_text { shift->_accept_targets('force_resolve', @_) }
251 # forces them to be processed, even when there's no ":".
252
253sub _accept_targets {
254 my($this, $type) = splice @_,0,2;
255 foreach my $t (@_) {
256 next unless defined $t and length $t;
257 # TODO: enforce some limitations on what a target name can be?
258 $this->{'accept_targets'}{$t} = $type;
259 DEBUG > 2 and print "Learning to accept \"$t\" as target of type $type\n";
260 }
261 return sort keys %{ $this->{'accept_targets'} } if wantarray;
262 return;
263}
264
265#--------------------------------------------------------------------------
266sub unaccept_target { shift->unaccept_targets(@_) }
267
268sub unaccept_targets {
269 my $this = shift;
270 foreach my $t (@_) {
271 next unless defined $t and length $t;
272 # TODO: enforce some limitations on what a target name can be?
273 delete $this->{'accept_targets'}{$t};
274 DEBUG > 2 and print "OK, won't accept \"$t\" as target.\n";
275 }
276 return sort keys %{ $this->{'accept_targets'} } if wantarray;
277 return;
278}
279
280#@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
281#
282# And now codes (not targets or directives)
283
284sub accept_code { shift->accept_codes(@_) } # alias
285
286sub accept_codes { # Add some codes
287 my $this = shift;
288
289 foreach my $new_code (@_) {
290 next unless defined $new_code and length $new_code;
291 if(ASCII) {
292 # A good-enough check that it's good as an XML Name symbol:
293 Carp::croak "\"$new_code\" isn't a valid element name"
294 if $new_code =~
295 m/[\x00-\x2C\x2F\x39\x3B-\x40\x5B-\x5E\x60\x7B-\x7F]/
296 # Characters under 0x80 that aren't legal in an XML Name.
297 or $new_code =~ m/^[-\.0-9]/s
298 or $new_code =~ m/:[-\.0-9]/s;
299 # The legal under-0x80 Name characters that
300 # an XML Name still can't start with.
301 }
302
303 $this->{'accept_codes'}{$new_code} = $new_code;
304
305 # Yes, map to itself -- just so that when we
306 # see "=extend W [whatever] thatelementname", we say that W maps
307 # to whatever $this->{accept_codes}{thatelementname} is,
308 # i.e., "thatelementname". Then when we go re-mapping,
309 # a "W" in the treelet turns into "thatelementname". We only
310 # remap once.
311 # If we say we accept "W", then a "W" in the treelet simply turns
312 # into "W".
313 }
314
315 return;
316}
317
318#--------------------------------------------------------------------------
319sub unaccept_code { shift->unaccept_codes(@_) }
320
321sub unaccept_codes { # remove some codes
322 my $this = shift;
323
324 foreach my $new_code (@_) {
325 next unless defined $new_code and length $new_code;
326 if(ASCII) {
327 # A good-enough check that it's good as an XML Name symbol:
328 Carp::croak "\"$new_code\" isn't a valid element name"
329 if $new_code =~
330 m/[\x00-\x2C\x2F\x39\x3B-\x40\x5B-\x5E\x60\x7B-\x7F]/
331 # Characters under 0x80 that aren't legal in an XML Name.
332 or $new_code =~ m/^[-\.0-9]/s
333 or $new_code =~ m/:[-\.0-9]/s;
334 # The legal under-0x80 Name characters that
335 # an XML Name still can't start with.
336 }
337
338 Carp::croak "But you must accept \"$new_code\" codes -- it's a builtin!"
339 if grep $new_code eq $_, @Known_formatting_codes;
340
341 delete $this->{'accept_codes'}{$new_code};
342
343 DEBUG > 2 and print "OK, won't accept the code $new_code<...>.\n";
344 }
345
346 return;
347}
348
349
350#@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
351#@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
352
353sub parse_string_document {
354 my $self = shift;
355 my @lines;
356 foreach my $line_group (@_) {
357 next unless defined $line_group and length $line_group;
358 pos($line_group) = 0;
359 while($line_group =~
360 m/([^\n\r]*)((?:\r?\n)?)/g
361 ) {
362 #print(">> $1\n"),
363 $self->parse_lines($1)
364 if length($1) or length($2)
365 or pos($line_group) != length($line_group);
366 # I.e., unless it's a zero-length "empty line" at the very
367 # end of "foo\nbar\n" (i.e., between the \n and the EOS).
368 }
369 }
370 $self->parse_lines(undef); # to signal EOF
371 return $self;
372}
373
374sub _init_fh_source {
375 my($self, $source) = @_;
376
377 #DEBUG > 1 and print "Declaring $source as :raw for starters\n";
378 #$self->_apply_binmode($source, ':raw');
379 #binmode($source, ":raw");
380
381 return;
382}
383
384#:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.
385#
386
387sub parse_file {
388 my($self, $source) = (@_);
389
390 if(!defined $source) {
391 Carp::croak("Can't use empty-string as a source for parse_file");
392 } elsif(ref(\$source) eq 'GLOB') {
393 $self->{'source_filename'} = '' . ($source);
394 } elsif(ref $source) {
395 $self->{'source_filename'} = '' . ($source);
396 } elsif(!length $source) {
397 Carp::croak("Can't use empty-string as a source for parse_file");
398 } else {
399 {
400 local *PODSOURCE;
401 open(PODSOURCE, "<$source") || Carp::croak("Can't open $source: $!");
402 $self->{'source_filename'} = $source;
403 $source = *PODSOURCE{IO};
404 }
405 $self->_init_fh_source($source);
406 }
407 # By here, $source is a FH.
408
409 $self->{'source_fh'} = $source;
410
411 my($i, @lines);
412 until( $self->{'source_dead'} ) {
413 splice @lines;
414 for($i = MANY_LINES; $i--;) { # read those many lines at a time
415 local $/ = $NL;
416 push @lines, scalar(<$source>); # readline
417 last unless defined $lines[-1];
418 # but pass thru the undef, which will set source_dead to true
419 }
420 $self->parse_lines(@lines);
421 }
422 delete($self->{'source_fh'}); # so it can be GC'd
423 return $self;
424}
425
426#:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.
427
428sub parse_from_file {
429 # An emulation of Pod::Parser's interface, for the sake of Perldoc.
430 # Basically just a wrapper around parse_file.
431
432 my($self, $source, $to) = @_;
433 $self = $self->new unless ref($self); # so we tolerate being a class method
434
435 if(!defined $source) { $source = *STDIN{IO}
436 } elsif(ref(\$source) eq 'GLOB') { # stet
437 } elsif(ref($source) ) { # stet
438 } elsif(!length $source
439 or $source eq '-' or $source =~ m/^<&(STDIN|0)$/i
440 ) {
441 $source = *STDIN{IO};
442 }
443
444 if(!defined $to) { $self->output_fh( *STDOUT{IO} );
445 } elsif(ref(\$to) eq 'GLOB') { $self->output_fh( $to );
446 } elsif(ref($to)) { $self->output_fh( $to );
447 } elsif(!length $to
448 or $to eq '-' or $to =~ m/^>&?(?:STDOUT|1)$/i
449 ) {
450 $self->output_fh( *STDOUT{IO} );
451 } else {
452 require Symbol;
453 my $out_fh = Symbol::gensym();
454 DEBUG and print "Write-opening to $to\n";
455 open($out_fh, ">$to") or Carp::croak "Can't write-open $to: $!";
456 binmode($out_fh)
457 if $self->can('write_with_binmode') and $self->write_with_binmode;
458 $self->output_fh($out_fh);
459 }
460
461 return $self->parse_file($source);
462}
463
464#-----------------------------------------------------------------------------
465
466sub whine {
467 #my($self,$line,$complaint) = @_;
468 my $self = shift(@_);
469 ++$self->{'errors_seen'};
470 if($self->{'no_whining'}) {
471 DEBUG > 9 and print "Discarding complaint (at line $_[0]) $_[1]\n because no_whining is on.\n";
472 return;
473 }
474 return $self->_complain_warn(@_) if $self->{'complain_stderr'};
475 return $self->_complain_errata(@_);
476}
477
478sub scream { # like whine, but not suppressable
479 #my($self,$line,$complaint) = @_;
480 my $self = shift(@_);
481 ++$self->{'errors_seen'};
482 return $self->_complain_warn(@_) if $self->{'complain_stderr'};
483 return $self->_complain_errata(@_);
484}
485
486sub _complain_warn {
487 my($self,$line,$complaint) = @_;
488 return printf STDERR "%s around line %s: %s\n",
489 $self->{'source_filename'} || 'Pod input', $line, $complaint;
490}
491
492sub _complain_errata {
493 my($self,$line,$complaint) = @_;
494 if( $self->{'no_errata_section'} ) {
495 DEBUG > 9 and print "Discarding erratum (at line $line) $complaint\n because no_errata_section is on.\n";
496 } else {
497 DEBUG > 9 and print "Queuing erratum (at line $line) $complaint\n";
498 push @{$self->{'errata'}{$line}}, $complaint
499 # for a report to be generated later!
500 }
501 return 1;
502}
503
504#@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
505
506sub _get_initial_item_type {
507 # A hack-wrapper here for when you have like "=over\n\n=item 456\n\n"
508 my($self, $para) = @_;
509 return $para->[1]{'~type'} if $para->[1]{'~type'};
510
511 return $para->[1]{'~type'} = 'text'
512 if join("\n", @{$para}[2 .. $#$para]) =~ m/^\s*(\d+)\.?\s*$/s and $1 ne '1';
513 # Else fall thru to the general case:
514 return $self->_get_item_type($para);
515}
516
517
518
519sub _get_item_type { # mutates the item!!
520 my($self, $para) = @_;
521 return $para->[1]{'~type'} if $para->[1]{'~type'};
522
523
524 # Otherwise we haven't yet been to this node. Maybe alter it...
525
526 my $content = join "\n", @{$para}[2 .. $#$para];
527
528 if($content =~ m/^\s*\*\s*$/s or $content =~ m/^\s*$/s) {
529 # Like: "=item *", "=item * ", "=item"
530 splice @$para, 2; # so it ends up just being ['=item', { attrhash } ]
531 $para->[1]{'~orig_content'} = $content;
532 return $para->[1]{'~type'} = 'bullet';
533
534 } elsif($content =~ m/^\s*\*\s+(.+)/s) { # tolerance
535
536 # Like: "=item * Foo bar baz";
537 $para->[1]{'~orig_content'} = $content;
538 $para->[1]{'~_freaky_para_hack'} = $1;
539 DEBUG > 2 and print " Tolerating $$para[2] as =item *\\n\\n$1\n";
540 splice @$para, 2; # so it ends up just being ['=item', { attrhash } ]
541 return $para->[1]{'~type'} = 'bullet';
542
543 } elsif($content =~ m/^\s*(\d+)\.?\s*$/s) {
544 # Like: "=item 1.", "=item 123412"
545
546 $para->[1]{'~orig_content'} = $content;
547 $para->[1]{'number'} = $1; # Yes, stores the number there!
548
549 splice @$para, 2; # so it ends up just being ['=item', { attrhash } ]
550 return $para->[1]{'~type'} = 'number';
551
552 } else {
553 # It's anything else.
554 return $para->[1]{'~type'} = 'text';
555
556 }
557}
558
559#-----------------------------------------------------------------------------
560
561sub _make_treelet {
562 my $self = shift; # and ($para, $start_line)
563 my $treelet;
564 if(!@_) {
565 return [''];
566 } if(ref $_[0] and ref $_[0][0] and $_[0][0][0] eq '~Top') {
567 # Hack so we can pass in fake-o pre-cooked paragraphs:
568 # just have the first line be a reference to a ['~Top', {}, ...]
569 # We use this feechure in gen_errata and stuff.
570
571 DEBUG and print "Applying precooked treelet hack to $_[0][0]\n";
572 $treelet = $_[0][0];
573 splice @$treelet, 0, 2; # lop the top off
574 return $treelet;
575 } else {
576 $treelet = $self->_treelet_from_formatting_codes(@_);
577 }
578
579 if( $self->_remap_sequences($treelet) ) {
580 $self->_treat_Zs($treelet); # Might as well nix these first
581 $self->_treat_Ls($treelet); # L has to precede E and S
582 $self->_treat_Es($treelet);
583 $self->_treat_Ss($treelet); # S has to come after E
584
585 $self->_wrap_up($treelet); # Nix X's and merge texties
586
587 } else {
588 DEBUG and print "Formatless treelet gets fast-tracked.\n";
589 # Very common case!
590 }
591
592 splice @$treelet, 0, 2; # lop the top off
593
594 return $treelet;
595}
596
597#:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.
598
599sub _wrap_up {
600 my($self, @stack) = @_;
601 my $nixx = $self->{'nix_X_codes'};
602 my $merge = $self->{'merge_text' };
603 return unless $nixx or $merge;
604
605 DEBUG > 2 and print "\nStarting _wrap_up traversal.\n",
606 $merge ? (" Merge mode on\n") : (),
607 $nixx ? (" Nix-X mode on\n") : (),
608 ;
609
610
611 my($i, $treelet);
612 while($treelet = shift @stack) {
613 DEBUG > 3 and print " Considering children of this $treelet->[0] node...\n";
614 for($i = 2; $i < @$treelet; ++$i) { # iterate over children
615 DEBUG > 3 and print " Considering child at $i ", pretty($treelet->[$i]), "\n";
616 if($nixx and ref $treelet->[$i] and $treelet->[$i][0] eq 'X') {
617 DEBUG > 3 and print " Nixing X node at $i\n";
618 splice(@$treelet, $i, 1); # just nix this node (and its descendants)
619 # no need to back-update the counter just yet
620 redo;
621
622 } elsif($merge and $i != 2 and # non-initial
623 !ref $treelet->[$i] and !ref $treelet->[$i - 1]
624 ) {
625 DEBUG > 3 and print " Merging ", $i-1,
626 ":[$treelet->[$i-1]] and $i\:[$treelet->[$i]]\n";
627 $treelet->[$i-1] .= ( splice(@$treelet, $i, 1) )[0];
628 DEBUG > 4 and print " Now: ", $i-1, ":[$treelet->[$i-1]]\n";
629 --$i;
630 next;
631 # since we just pulled the possibly last node out from under
632 # ourselves, we can't just redo()
633
634 } elsif( ref $treelet->[$i] ) {
635 DEBUG > 4 and print " Enqueuing ", pretty($treelet->[$i]), " for traversal.\n";
636 push @stack, $treelet->[$i];
637
638 if($treelet->[$i][0] eq 'L') {
639 my $thing;
640 foreach my $attrname ('section', 'to') {
641 if(defined($thing = $treelet->[$i][1]{$attrname}) and ref $thing) {
642 unshift @stack, $thing;
643 DEBUG > 4 and print " +Enqueuing ",
644 pretty( $treelet->[$i][1]{$attrname} ),
645 " as an attribute value to tweak.\n";
646 }
647 }
648 }
649 }
650 }
651 }
652 DEBUG > 2 and print "End of _wrap_up traversal.\n\n";
653
654 return;
655}
656
657#:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.
658
659sub _remap_sequences {
660 my($self,@stack) = @_;
661
662 if(@stack == 1 and @{ $stack[0] } == 3 and !ref $stack[0][2]) {
663 # VERY common case: abort it.
664 DEBUG and print "Skipping _remap_sequences: formatless treelet.\n";
665 return 0;
666 }
667
668 my $map = ($self->{'accept_codes'} || die "NO accept_codes in $self?!?");
669
670 my $start_line = $stack[0][1]{'start_line'};
671 DEBUG > 2 and printf
672 "\nAbout to start _remap_sequences on treelet from line %s.\n",
673 $start_line || '[?]'
674 ;
675 DEBUG > 3 and print " Map: ",
676 join('; ', map "$_=" . (
677 ref($map->{$_}) ? join(",", @{$map->{$_}}) : $map->{$_}
678 ),
679 sort keys %$map ),
680 ("B~C~E~F~I~L~S~X~Z" eq join '~', sort keys %$map)
681 ? " (all normal)\n" : "\n"
682 ;
683
684 # A recursive algorithm implemented iteratively! Whee!
685
686 my($is, $was, $i, $treelet); # scratch
687 while($treelet = shift @stack) {
688 DEBUG > 3 and print " Considering children of this $treelet->[0] node...\n";
689 for($i = 2; $i < @$treelet; ++$i) { # iterate over children
690 next unless ref $treelet->[$i]; # text nodes are uninteresting
691
692 DEBUG > 4 and print " Noting child $i : $treelet->[$i][0]<...>\n";
693
694 $is = $treelet->[$i][0] = $map->{ $was = $treelet->[$i][0] };
695 if( DEBUG > 3 ) {
696 if(!defined $is) {
697 print " Code $was<> is UNKNOWN!\n";
698 } elsif($is eq $was) {
699 DEBUG > 4 and print " Code $was<> stays the same.\n";
700 } else {
701 print " Code $was<> maps to ",
702 ref($is)
703 ? ( "tags ", map("$_<", @$is), '...', map('>', @$is), "\n" )
704 : "tag $is<...>.\n";
705 }
706 }
707
708 if(!defined $is) {
709 $self->whine($start_line, "Deleting unknown formatting code $was<>");
710 $is = $treelet->[$i][0] = '1'; # But saving the children!
711 # I could also insert a leading "$was<" and tailing ">" as
712 # children of this node, but something about that seems icky.
713 }
714 if(ref $is) {
715 my @dynasty = @$is;
716 DEBUG > 4 and print " Renaming $was node to $dynasty[-1]\n";
717 $treelet->[$i][0] = pop @dynasty;
718 my $nugget;
719 while(@dynasty) {
720 DEBUG > 4 and printf
721 " Grafting a new %s node between %s and %s\n",
722 $dynasty[-1], $treelet->[0], $treelet->[$i][0],
723 ;
724
725 #$nugget = ;
726 splice @$treelet, $i, 1, [pop(@dynasty), {}, $treelet->[$i]];
727 # relace node with a new parent
728 }
729 } elsif($is eq '0') {
730 splice(@$treelet, $i, 1); # just nix this node (and its descendants)
731 --$i; # back-update the counter
732 } elsif($is eq '1') {
733 splice(@$treelet, $i, 1 # replace this node with its children!
734 => splice @{ $treelet->[$i] },2
735 # (not catching its first two (non-child) items)
736 );
737 --$i; # back up for new stuff
738 } else {
739 # otherwise it's unremarkable
740 unshift @stack, $treelet->[$i]; # just recurse
741 }
742 }
743 }
744
745 DEBUG > 2 and print "End of _remap_sequences traversal.\n\n";
746
747 if(@_ == 2 and @{ $_[1] } == 3 and !ref $_[1][2]) {
748 DEBUG and print "Noting that the treelet is now formatless.\n";
749 return 0;
750 }
751 return 1;
752}
753
754# . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
755
756sub _ponder_extend {
757
758 # "Go to an extreme, move back to a more comfortable place"
759 # -- /Oblique Strategies/, Brian Eno and Peter Schmidt
760
761 my($self, $para) = @_;
762 my $content = join ' ', splice @$para, 2;
763 $content =~ s/^\s+//s;
764 $content =~ s/\s+$//s;
765
766 DEBUG > 2 and print "Ogling extensor: =extend $content\n";
767
768 if($content =~
769 m/^
770 (\S+) # 1 : new item
771 \s+
772 (\S+) # 2 : fallback(s)
773 (?:\s+(\S+))? # 3 : element name(s)
774 \s*
775 $
776 /xs
777 ) {
778 my $new_letter = $1;
779 my $fallbacks_one = $2;
780 my $elements_one;
781 $elements_one = defined($3) ? $3 : $1;
782
783 DEBUG > 2 and print "Extensor has good syntax.\n";
784
785 unless($new_letter =~ m/^[A-Z]$/s or $new_letter) {
786 DEBUG > 2 and print " $new_letter isn't a valid thing to entend.\n";
787 $self->whine(
788 $para->[1]{'start_line'},
789 "You can extend only formatting codes A-Z, not like \"$new_letter\""
790 );
791 return;
792 }
793
794 if(grep $new_letter eq $_, @Known_formatting_codes) {
795 DEBUG > 2 and print " $new_letter isn't a good thing to extend, because known.\n";
796 $self->whine(
797 $para->[1]{'start_line'},
798 "You can't extend an established code like \"$new_letter\""
799 );
800
801 #TODO: or allow if last bit is same?
802
803 return;
804 }
805
806 unless($fallbacks_one =~ m/^[A-Z](,[A-Z])*$/s # like "B", "M,I", etc.
807 or $fallbacks_one eq '0' or $fallbacks_one eq '1'
808 ) {
809 $self->whine(
810 $para->[1]{'start_line'},
811 "Format for second =extend parameter must be like"
812 . " M or 1 or 0 or M,N or M,N,O but you have it like "
813 . $fallbacks_one
814 );
815 return;
816 }
817
818 unless($elements_one =~ m/^[^ ,]+(,[^ ,]+)*$/s) { # like "B", "M,I", etc.
819 $self->whine(
820 $para->[1]{'start_line'},
821 "Format for third =extend parameter: like foo or bar,Baz,qu:ux but not like "
822 . $elements_one
823 );
824 return;
825 }
826
827 my @fallbacks = split ',', $fallbacks_one, -1;
828 my @elements = split ',', $elements_one, -1;
829
830 foreach my $f (@fallbacks) {
831 next if exists $Known_formatting_codes{$f} or $f eq '0' or $f eq '1';
832 DEBUG > 2 and print " Can't fall back on unknown code $f\n";
833 $self->whine(
834 $para->[1]{'start_line'},
835 "Can't use unknown formatting code '$f' as a fallback for '$new_letter'"
836 );
837 return;
838 }
839
840 DEBUG > 3 and printf "Extensor: Fallbacks <%s> Elements <%s>.\n",
841 @fallbacks, @elements;
842
843 my $canonical_form;
844 foreach my $e (@elements) {
845 if(exists $self->{'accept_codes'}{$e}) {
846 DEBUG > 1 and print " Mapping '$new_letter' to known extension '$e'\n";
847 $canonical_form = $e;
848 last; # first acceptable elementname wins!
849 } else {
850 DEBUG > 1 and print " Can't map '$new_letter' to unknown extension '$e'\n";
851 }
852 }
853
854
855 if( defined $canonical_form ) {
856 # We found a good N => elementname mapping
857 $self->{'accept_codes'}{$new_letter} = $canonical_form;
858 DEBUG > 2 and print
859 "Extensor maps $new_letter => known element $canonical_form.\n";
860 } else {
861 # We have to use the fallback(s), which might be '0', or '1'.
862 $self->{'accept_codes'}{$new_letter}
863 = (@fallbacks == 1) ? $fallbacks[0] : \@fallbacks;
864 DEBUG > 2 and print
865 "Extensor maps $new_letter => fallbacks @fallbacks.\n";
866 }
867
868 } else {
869 DEBUG > 2 and print "Extensor has bad syntax.\n";
870 $self->whine(
871 $para->[1]{'start_line'},
872 "Unknown =extend syntax: $content"
873 )
874 }
875 return;
876}
877
878
879#:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.:.
880
881sub _treat_Zs { # Nix Z<...>'s
882 my($self,@stack) = @_;
883
884 my($i, $treelet);
885 my $start_line = $stack[0][1]{'start_line'};
886
887 # A recursive algorithm implemented iteratively! Whee!
888
889 while($treelet = shift @stack) {
890 for($i = 2; $i < @$treelet; ++$i) { # iterate over children
891 next unless ref $treelet->[$i]; # text nodes are uninteresting
892 unless($treelet->[$i][0] eq 'Z') {
893 unshift @stack, $treelet->[$i]; # recurse
894 next;
895 }
896
897 DEBUG > 1 and print "Nixing Z node @{$treelet->[$i]}\n";
898
899 # bitch UNLESS it's empty
900 unless( @{$treelet->[$i]} == 2
901 or (@{$treelet->[$i]} == 3 and $treelet->[$i][2] eq '')
902 ) {
903 $self->whine( $start_line, "A non-empty Z<>" );
904 } # but kill it anyway
905
906 splice(@$treelet, $i, 1); # thereby just nix this node.
907 --$i;
908
909 }
910 }
911
912 return;
913}
914
915# . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
916
917# Quoting perlpodspec:
918
919# In parsing an L<...> code, Pod parsers must distinguish at least four
920# attributes:
921
922############# Not used. Expressed via the element children plus
923############# the value of the "content-implicit" flag.
924# First:
925# The link-text. If there is none, this must be undef. (E.g., in "L<Perl
926# Functions|perlfunc>", the link-text is "Perl Functions". In
927# "L<Time::HiRes>" and even "L<|Time::HiRes>", there is no link text. Note
928# that link text may contain formatting.)
929#
930
931############# The element children
932# Second:
933# The possibly inferred link-text -- i.e., if there was no real link text,
934# then this is the text that we'll infer in its place. (E.g., for
935# "L<Getopt::Std>", the inferred link text is "Getopt::Std".)
936#
937
938############# The "to" attribute (which might be text, or a treelet)
939# Third:
940# The name or URL, or undef if none. (E.g., in "L<Perl
941# Functions|perlfunc>", the name -- also sometimes called the page -- is
942# "perlfunc". In "L</CAVEATS>", the name is undef.)
943#
944
945############# The "section" attribute (which might be next, or a treelet)
946# Fourth:
947# The section (AKA "item" in older perlpods), or undef if none. E.g., in
948# Getopt::Std/DESCRIPTION, "DESCRIPTION" is the section. (Note that this
949# is not the same as a manpage section like the "5" in "man 5 crontab".
950# "Section Foo" in the Pod sense means the part of the text that's
951# introduced by the heading or item whose text is "Foo".)
952#
953# Pod parsers may also note additional attributes including:
954#
955
956############# The "type" attribute.
957# Fifth:
958# A flag for whether item 3 (if present) is a URL (like
959# "http://lists.perl.org" is), in which case there should be no section
960# attribute; a Pod name (like "perldoc" and "Getopt::Std" are); or
961# possibly a man page name (like "crontab(5)" is).
962#
963
964############# Not implemented, I guess.
965# Sixth:
966# The raw original L<...> content, before text is split on "|", "/", etc,
967# and before E<...> codes are expanded.
968
969
970# For L<...> codes without a "name|" part, only E<...> and Z<> codes may
971# occur -- no other formatting codes. That is, authors should not use
972# "L<B<Foo::Bar>>".
973#
974# Note, however, that formatting codes and Z<>'s can occur in any and all
975# parts of an L<...> (i.e., in name, section, text, and url).
976
977sub _treat_Ls { # Process our dear dear friends, the L<...> sequences
978
979 # L<name>
980 # L<name/"sec"> or L<name/sec>
981 # L</"sec"> or L</sec> or L<"sec">
982 # L<text|name>
983 # L<text|name/"sec"> or L<text|name/sec>
984 # L<text|/"sec"> or L<text|/sec> or L<text|"sec">
985 # L<scheme:...>
986
987 my($self,@stack) = @_;
988
989 my($i, $treelet);
990 my $start_line = $stack[0][1]{'start_line'};
991
992 # A recursive algorithm implemented iteratively! Whee!
993
994 while($treelet = shift @stack) {
995 for(my $i = 2; $i < @$treelet; ++$i) {
996 # iterate over children of current tree node
997 next unless ref $treelet->[$i]; # text nodes are uninteresting
998 unless($treelet->[$i][0] eq 'L') {
999 unshift @stack, $treelet->[$i]; # recurse
1000 next;
1001 }
1002
1003
1004 # By here, $treelet->[$i] is definitely an L node
1005 DEBUG > 1 and print "Ogling L node $treelet->[$i]\n";
1006
1007 # bitch if it's empty
1008 if( @{$treelet->[$i]} == 2
1009 or (@{$treelet->[$i]} == 3 and $treelet->[$i][2] eq '')
1010 ) {
1011 $self->whine( $start_line, "An empty L<>" );
1012 $treelet->[$i] = 'L<>'; # just make it a text node
1013 next; # and move on
1014 }
1015
1016 # Catch URLs:
1017 # URLs can, alas, contain E<...> sequences, so we can't /assume/
1018 # that this is one text node. But it has to START with one text
1019 # node...
1020 if(! ref $treelet->[$i][2] and
1021 $treelet->[$i][2] =~ m/^\w+:[^:\s]\S*$/s
1022 ) {
1023 $treelet->[$i][1]{'type'} = 'url';
1024 $treelet->[$i][1]{'content-implicit'} = 'yes';
1025
1026 # TODO: deal with rel: URLs here?
1027
1028 if( 3 == @{ $treelet->[$i] } ) {
1029 # But if it IS just one text node (most common case)
1030 DEBUG > 1 and printf qq{Catching "%s as " as ho-hum L<URL> link.\n},
1031 $treelet->[$i][2]
1032 ;
1033 $treelet->[$i][1]{'to'} = Pod::Simple::LinkSection->new(
1034 $treelet->[$i][2]
1035 ); # its own treelet
1036 } else {
1037 # It's a URL but complex (like "L<foo:bazE<123>bar>"). Feh.
1038 #$treelet->[$i][1]{'to'} = [ @{$treelet->[$i]} ];
1039 #splice @{ $treelet->[$i][1]{'to'} }, 0,2;
1040 #DEBUG > 1 and printf qq{Catching "%s as " as complex L<URL> link.\n},
1041 # join '~', @{$treelet->[$i][1]{'to' }};
1042
1043 $treelet->[$i][1]{'to'} = Pod::Simple::LinkSection->new(
1044 $treelet->[$i] # yes, clone the whole content as a treelet
1045 );
1046 $treelet->[$i][1]{'to'}[0] = ''; # set the copy's tagname to nil
1047 die "SANITY FAILURE" if $treelet->[0] eq ''; # should never happen!
1048 DEBUG > 1 and print
1049 qq{Catching "$treelet->[$i][1]{'to'}" as a complex L<URL> link.\n};
1050 }
1051
1052 next; # and move on
1053 }
1054
1055
1056 # Catch some very simple and/or common cases
1057 if(@{$treelet->[$i]} == 3 and ! ref $treelet->[$i][2]) {
1058 my $it = $treelet->[$i][2];
1059 if($it =~ m/^[-a-zA-Z0-9]+\([-a-zA-Z0-9]+\)$/s) { # man sections
1060 # Hopefully neither too broad nor too restrictive a RE
1061 DEBUG > 1 and print "Catching \"$it\" as manpage link.\n";
1062 $treelet->[$i][1]{'type'} = 'man';
1063 # This's the only place where man links can get made.
1064 $treelet->[$i][1]{'content-implicit'} = 'yes';
1065 $treelet->[$i][1]{'to' } =
1066 Pod::Simple::LinkSection->new( $it ); # treelet!
1067
1068 next;
1069 }
1070 if($it =~ m/^[^\/\|,\$\%\@\ \"\<\>\:\#\&\*\{\}\[\]\(\)]+(\:\:[^\/\|,\$\%\@\ \"\<\>\:\#\&\*\{\}\[\]\(\)]+)*$/s) {
1071 # Extremely forgiving idea of what constitutes a bare
1072 # modulename link like L<Foo::Bar> or even L<Thing::1.0::Docs::Tralala>
1073 DEBUG > 1 and print "Catching \"$it\" as ho-hum L<Modulename> link.\n";
1074 $treelet->[$i][1]{'type'} = 'pod';
1075 $treelet->[$i][1]{'content-implicit'} = 'yes';
1076 $treelet->[$i][1]{'to' } =
1077 Pod::Simple::LinkSection->new( $it ); # treelet!
1078 next;
1079 }
1080 # else fall thru...
1081 }
1082
1083
1084
1085 # ...Uhoh, here's the real L<...> parsing stuff...
1086 # "With the ill behavior, with the ill behavior, with the ill behavior..."
1087
1088 DEBUG > 1 and print "Running a real parse on this non-trivial L\n";
1089
1090
1091 my $link_text; # set to an arrayref if found
1092 my $ell = $treelet->[$i];
1093 my @ell_content = @$ell;
1094 splice @ell_content,0,2; # Knock off the 'L' and {} bits
1095
1096 DEBUG > 3 and print " Ell content to start: ",
1097 pretty(@ell_content), "\n";
1098
1099
1100 # Look for the "|" -- only in CHILDREN (not all underlings!)
1101 # Like L<I like the strictness|strict>
1102 DEBUG > 3 and
1103 print " Peering at L content for a '|' ...\n";
1104 for(my $j = 0; $j < @ell_content; ++$j) {
1105 next if ref $ell_content[$j];
1106 DEBUG > 3 and
1107 print " Peering at L-content text bit \"$ell_content[$j]\" for a '|'.\n";
1108
1109 if($ell_content[$j] =~ m/^([^\|]*)\|(.*)$/s) {
1110 my @link_text = ($1); # might be 0-length
1111 $ell_content[$j] = $2; # might be 0-length
1112
1113 DEBUG > 3 and
1114 print " FOUND a '|' in it. Splitting into [$1] + [$2]\n";
1115
1116 unshift @link_text, splice @ell_content, 0, $j;
1117 # leaving only things at J and after
1118 @ell_content = grep ref($_)||length($_), @ell_content ;
1119 $link_text = [grep ref($_)||length($_), @link_text ];
1120 DEBUG > 3 and printf
1121 " So link text is %s\n and remaining ell content is %s\n",
1122 pretty($link_text), pretty(@ell_content);
1123 last;
1124 }
1125 }
1126
1127
1128 # Now look for the "/" -- only in CHILDREN (not all underlings!)
1129 # And afterward, anything left in @ell_content will be the raw name
1130 # Like L<Foo::Bar/Object Methods>
1131 my $section_name; # set to arrayref if found
1132 DEBUG > 3 and print " Peering at L-content for a '/' ...\n";
1133 for(my $j = 0; $j < @ell_content; ++$j) {
1134 next if ref $ell_content[$j];
1135 DEBUG > 3 and
1136 print " Peering at L-content text bit \"$ell_content[$j]\" for a '/'.\n";
1137
1138 if($ell_content[$j] =~ m/^([^\/]*)\/(.*)$/s) {
1139 my @section_name = ($2); # might be 0-length
1140 $ell_content[$j] = $1; # might be 0-length
1141
1142 DEBUG > 3 and
1143 print " FOUND a '/' in it.",
1144 " Splitting to page [...$1] + section [$2...]\n";
1145
1146 push @section_name, splice @ell_content, 1+$j;
1147 # leaving only things before and including J
1148
1149 @ell_content = grep ref($_)||length($_), @ell_content ;
1150 @section_name = grep ref($_)||length($_), @section_name ;
1151
1152 # Turn L<.../"foo"> into L<.../foo>
1153 if(@section_name
1154 and !ref($section_name[0]) and !ref($section_name[-1])
1155 and $section_name[ 0] =~ m/^\"/s
1156 and $section_name[-1] =~ m/\"$/s
1157 and !( # catch weird degenerate case of L<"> !
1158 @section_name == 1 and $section_name[0] eq '"'
1159 )
1160 ) {
1161 $section_name[ 0] =~ s/^\"//s;
1162 $section_name[-1] =~ s/\"$//s;
1163 DEBUG > 3 and
1164 print " Quotes removed: ", pretty(@section_name), "\n";
1165 } else {
1166 DEBUG > 3 and
1167 print " No need to remove quotes in ", pretty(@section_name), "\n";
1168 }
1169
1170 $section_name = \@section_name;
1171 last;
1172 }
1173 }
1174
1175 # Turn L<"Foo Bar"> into L</Foo Bar>
1176 if(!$section_name and @ell_content
1177 and !ref($ell_content[0]) and !ref($ell_content[-1])
1178 and $ell_content[ 0] =~ m/^\"/s
1179 and $ell_content[-1] =~ m/\"$/s
1180 and !( # catch weird degenerate case of L<"> !
1181 @ell_content == 1 and $ell_content[0] eq '"'
1182 )
1183 ) {
1184 $section_name = [splice @ell_content];
1185 $section_name->[ 0] =~ s/^\"//s;
1186 $section_name->[-1] =~ s/\"$//s;
1187 }
1188
1189 # Turn L<Foo Bar> into L</Foo Bar>.
1190 if(!$section_name and !$link_text and @ell_content
1191 and grep !ref($_) && m/ /s, @ell_content
1192 ) {
1193 $section_name = [splice @ell_content];
1194 # That's support for the now-deprecated syntax.
1195 # (Maybe generate a warning eventually?)
1196 # Note that it deliberately won't work on L<...|Foo Bar>
1197 }
1198
1199
1200 # Now make up the link_text
1201 # L<Foo> -> L<Foo|Foo>
1202 # L</Bar> -> L<"Bar"|Bar>
1203 # L<Foo/Bar> -> L<"Bar" in Foo/Foo>
1204 unless($link_text) {
1205 $ell->[1]{'content-implicit'} = 'yes';
1206 $link_text = [];
1207 push @$link_text, '"', @$section_name, '"' if $section_name;
1208
1209 if(@ell_content) {
1210 $link_text->[-1] .= ' in ' if $section_name;
1211 push @$link_text, @ell_content;
1212 }
1213 }
1214
1215
1216 # And the E resolver will have to deal with all our treeletty things:
1217
1218 if(@ell_content == 1 and !ref($ell_content[0])
1219 and $ell_content[0] =~ m/^[-a-zA-Z0-9]+\([-a-zA-Z0-9]+\)$/s
1220 ) {
1221 $ell->[1]{'type'} = 'man';
1222 DEBUG > 3 and print "Considering this ($ell_content[0]) a man link.\n";
1223 } else {
1224 $ell->[1]{'type'} = 'pod';
1225 DEBUG > 3 and print "Considering this a pod link (not man or url).\n";
1226 }
1227
1228 if( defined $section_name ) {
1229 $ell->[1]{'section'} = Pod::Simple::LinkSection->new(
1230 ['', {}, @$section_name]
1231 );
1232 DEBUG > 3 and print "L-section content: ", pretty($ell->[1]{'section'}), "\n";
1233 }
1234
1235 if( @ell_content ) {
1236 $ell->[1]{'to'} = Pod::Simple::LinkSection->new(
1237 ['', {}, @ell_content]
1238 );
1239 DEBUG > 3 and print "L-to content: ", pretty($ell->[1]{'to'}), "\n";
1240 }
1241
1242 # And update children to be the link-text:
1243 @$ell = (@$ell[0,1], defined($link_text) ? splice(@$link_text) : '');
1244
1245 DEBUG > 2 and print "End of L-parsing for this node $treelet->[$i]\n";
1246
1247 unshift @stack, $treelet->[$i]; # might as well recurse
1248 }
1249 }
1250
1251 return;
1252}
1253
1254# . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
1255
1256sub _treat_Es {
1257 my($self,@stack) = @_;
1258
1259 my($i, $treelet, $content, $replacer, $charnum);
1260 my $start_line = $stack[0][1]{'start_line'};
1261
1262 # A recursive algorithm implemented iteratively! Whee!
1263
1264
1265 # Has frightening side effects on L nodes' attributes.
1266
1267 #my @ells_to_tweak;
1268
1269 while($treelet = shift @stack) {
1270 for(my $i = 2; $i < @$treelet; ++$i) { # iterate over children
1271 next unless ref $treelet->[$i]; # text nodes are uninteresting
1272 if($treelet->[$i][0] eq 'L') {
1273 # SPECIAL STUFF for semi-processed L<>'s
1274
1275 my $thing;
1276 foreach my $attrname ('section', 'to') {
1277 if(defined($thing = $treelet->[$i][1]{$attrname}) and ref $thing) {
1278 unshift @stack, $thing;
1279 DEBUG > 2 and print " Enqueuing ",
1280 pretty( $treelet->[$i][1]{$attrname} ),
1281 " as an attribute value to tweak.\n";
1282 }
1283 }
1284
1285 unshift @stack, $treelet->[$i]; # recurse
1286 next;
1287 } elsif($treelet->[$i][0] ne 'E') {
1288 unshift @stack, $treelet->[$i]; # recurse
1289 next;
1290 }
1291
1292 DEBUG > 1 and print "Ogling E node ", pretty($treelet->[$i]), "\n";
1293
1294 # bitch if it's empty
1295 if( @{$treelet->[$i]} == 2
1296 or (@{$treelet->[$i]} == 3 and $treelet->[$i][2] eq '')
1297 ) {
1298 $self->whine( $start_line, "An empty E<>" );
1299 $treelet->[$i] = 'E<>'; # splice in a literal
1300 next;
1301 }
1302
1303 # bitch if content is weird
1304 unless(@{$treelet->[$i]} == 3 and !ref($content = $treelet->[$i][2])) {
1305 $self->whine( $start_line, "An E<...> surrounding strange content" );
1306 $replacer = $treelet->[$i]; # scratch
1307 splice(@$treelet, $i, 1, # fake out a literal
1308 'E<',
1309 splice(@$replacer,2), # promote its content
1310 '>'
1311 );
1312 # Don't need to do --$i, as the 'E<' we just added isn't interesting.
1313 next;
1314 }
1315
1316 DEBUG > 1 and print "Ogling E<$content>\n";
1317
1318 $charnum = Pod::Escapes::e2charnum($content);
1319 DEBUG > 1 and print " Considering E<$content> with char ",
1320 defined($charnum) ? $charnum : "undef", ".\n";
1321
1322 if(!defined( $charnum )) {
1323 DEBUG > 1 and print "I don't know how to deal with E<$content>.\n";
1324 $self->whine( $start_line, "Unknown E content in E<$content>" );
1325 $replacer = "E<$content>"; # better than nothing
1326 } elsif($charnum >= 255 and !UNICODE) {
1327 $replacer = ASCII ? "\xA4" : "?";
1328 DEBUG > 1 and print "This Perl version can't handle ",
1329 "E<$content> (chr $charnum), so replacing with $replacer\n";
1330 } else {
1331 $replacer = Pod::Escapes::e2char($content);
1332 DEBUG > 1 and print " Replacing E<$content> with $replacer\n";
1333 }
1334
1335 splice(@$treelet, $i, 1, $replacer); # no need to back up $i, tho
1336 }
1337 }
1338
1339 return;
1340}
1341
1342
1343# . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
1344
1345sub _treat_Ss {
1346 my($self,$treelet) = @_;
1347
1348 _change_S_to_nbsp($treelet,0) if $self->{'nbsp_for_S'};
1349
1350 # TODO: or a change_nbsp_to_S
1351 # Normalizing nbsp's to S is harder: for each text node, make S content
1352 # out of anything matching m/([^ \xA0]*(?:\xA0+[^ \xA0]*)+)/
1353
1354
1355 return;
1356}
1357
1358
1359sub _change_S_to_nbsp { # a recursive function
1360 # Sanely assumes that the top node in the excursion won't be an S node.
1361 my($treelet, $in_s) = @_;
1362
1363 my $is_s = ('S' eq $treelet->[0]);
1364 $in_s ||= $is_s; # So in_s is on either by this being an S element,
1365 # or by an ancestor being an S element.
1366
1367 for(my $i = 2; $i < @$treelet; ++$i) {
1368 if(ref $treelet->[$i]) {
1369 if( _change_S_to_nbsp( $treelet->[$i], $in_s ) ) {
1370 my $to_pull_up = $treelet->[$i];
1371 splice @$to_pull_up,0,2; # ...leaving just its content
1372 splice @$treelet, $i, 1, @$to_pull_up; # Pull up content
1373 $i += @$to_pull_up - 1; # Make $i skip the pulled-up stuff
1374 }
1375 } else {
1376 $treelet->[$i] =~ s/\s/\xA0/g if ASCII and $in_s;
1377 # (If not in ASCIIland, we can't assume that \xA0 == nbsp.)
1378
1379 # Note that if you apply nbsp_for_S to text, and so turn
1380 # "foo S<bar baz> quux" into "foo bar&#160;faz quux", you
1381 # end up with something that fails to say "and don't hyphenate
1382 # any part of 'bar baz'". However, hyphenation is such a vexing
1383 # problem anyway, that most Pod renderers just don't render it
1384 # at all. But if you do want to implement hyphenation, I guess
1385 # that you'd better have nbsp_for_S off.
1386 }
1387 }
1388
1389 return $is_s;
1390}
1391
1392#-----------------------------------------------------------------------------
1393
1394sub _accessorize { # A simple-minded method-maker
1395 no strict 'refs';
1396 foreach my $attrname (@_) {
1397 next if $attrname =~ m/::/; # a hack
1398 *{caller() . '::' . $attrname} = sub {
1399 use strict;
1400 $Carp::CarpLevel = 1, Carp::croak(
1401 "Accessor usage: \$obj->$attrname() or \$obj->$attrname(\$new_value)"
1402 ) unless (@_ == 1 or @_ == 2) and ref $_[0];
1403 (@_ == 1) ? $_[0]->{$attrname}
1404 : ($_[0]->{$attrname} = $_[1]);
1405 };
1406 }
1407 # Ya know, they say accessories make the ensemble!
1408 return;
1409}
1410
1411# . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
1412# . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . .
1413#=============================================================================
1414
1415sub filter {
1416 my($class, $source) = @_;
1417 my $new = $class->new;
1418 $new->output_fh(*STDOUT{IO});
1419
1420 if(ref($source || '') eq 'SCALAR') {
1421 $new->parse_string_document( $$source );
1422 } elsif(ref($source)) { # it's a file handle
1423 $new->parse_file($source);
1424 } else { # it's a filename
1425 $new->parse_file($source);
1426 }
1427
1428 return $new;
1429}
1430
1431
1432#-----------------------------------------------------------------------------
1433
1434sub _out {
1435 # For use in testing: Class->_out($source)
1436 # returns the transformation of $source
1437
1438 my $class = shift(@_);
1439
1440 my $mutor = shift(@_) if @_ and ref($_[0] || '') eq 'CODE';
1441
1442 DEBUG and print "\n\n", '#' x 76,
1443 "\nAbout to parse source: {{\n$_[0]\n}}\n\n";
1444
1445
1446 my $parser = $class->new;
1447 $parser->hide_line_numbers(1);
1448
1449 my $out = '';
1450 $parser->output_string( \$out );
1451 DEBUG and print " _out to ", \$out, "\n";
1452
1453 $mutor->($parser) if $mutor;
1454
1455 $parser->parse_string_document( $_[0] );
1456 # use Data::Dumper; print Dumper($parser), "\n";
1457 return $out;
1458}
1459
1460
1461sub _duo {
1462 # For use in testing: Class->_duo($source1, $source2)
1463 # returns the parse trees of $source1 and $source2.
1464 # Good in things like: &ok( Class->duo(... , ...) );
1465
1466 my $class = shift(@_);
1467
1468 Carp::croak "But $class->_duo is useful only in list context!"
1469 unless wantarray;
1470
1471 my $mutor = shift(@_) if @_ and ref($_[0] || '') eq 'CODE';
1472
1473 Carp::croak "But $class->_duo takes two parameters, not: @_"
1474 unless @_ == 2;
1475
1476 my(@out);
1477
1478 while( @_ ) {
1479 my $parser = $class->new;
1480
1481 push @out, '';
1482 $parser->output_string( \( $out[-1] ) );
1483
1484 DEBUG and print " _duo out to ", $parser->output_string(),
1485 " = $parser->{'output_string'}\n";
1486
1487 $parser->hide_line_numbers(1);
1488 $mutor->($parser) if $mutor;
1489 $parser->parse_string_document( shift( @_ ) );
1490 # use Data::Dumper; print Dumper($parser), "\n";
1491 }
1492
1493 return @out;
1494}
1495
1496
1497
1498#-----------------------------------------------------------------------------
14991;
1500__END__
1501
1502TODO:
1503A start_formatting_code and end_formatting_code methods, which in the
1504base class call start_L, end_L, start_C, end_C, etc., if they are
1505defined.
1506
1507have the POD FORMATTING ERRORS section note the localtime, and the
1508version of Pod::Simple.
1509
1510option to delete all E<shy>s?
1511option to scream if under-0x20 literals are found in the input, or
1512under-E<32> E codes are found in the tree. And ditto \x7f-\x9f
1513
1514Option to turn highbit characters into their compromised form? (applies
1515to E parsing too)
1516
1517TODO: BOM/encoding things.
1518
1519TODO: ascii-compat things in the XML classes?
1520