| line | stmt | bran | cond | sub | pod | time | code | 
| 1 |  |  |  |  |  |  | # Text::Aligner - Align text in columns | 
| 2 |  |  |  |  |  |  | package Text::Aligner; | 
| 3 |  |  |  |  |  |  |  | 
| 4 | 2 |  |  | 2 |  | 69299 | use strict; | 
|  | 2 |  |  |  |  | 19 |  | 
|  | 2 |  |  |  |  | 62 |  | 
| 5 | 2 |  |  | 2 |  | 11 | use warnings; | 
|  | 2 |  |  |  |  | 4 |  | 
|  | 2 |  |  |  |  | 54 |  | 
| 6 |  |  |  |  |  |  |  | 
| 7 | 2 |  |  | 2 |  | 49 | use 5.008; | 
|  | 2 |  |  |  |  | 7 |  | 
| 8 |  |  |  |  |  |  |  | 
| 9 |  |  |  |  |  |  | our $VERSION = '0.13'; | 
| 10 |  |  |  |  |  |  |  | 
| 11 |  |  |  |  |  |  | BEGIN    { | 
| 12 | 2 |  |  | 2 |  | 11 | use Exporter (); | 
|  | 2 |  |  |  |  | 3 |  | 
|  | 2 |  |  |  |  | 77 |  | 
| 13 | 2 |  |  | 2 |  | 11 | use vars qw (@ISA @EXPORT @EXPORT_OK %EXPORT_TAGS); | 
|  | 2 |  |  |  |  | 4 |  | 
|  | 2 |  |  |  |  | 258 |  | 
| 14 | 2 |  |  | 2 |  | 49 | @ISA         = qw (Exporter); | 
| 15 | 2 |  |  |  |  | 8 | @EXPORT      = qw (); | 
| 16 | 2 |  |  |  |  | 4 | @EXPORT_OK   = qw ( align); | 
| 17 | 2 |  |  |  |  | 1059 | %EXPORT_TAGS = (); | 
| 18 |  |  |  |  |  |  | } | 
| 19 |  |  |  |  |  |  |  | 
| 20 |  |  |  |  |  |  | # this is a non-method, and currently the only user interface | 
| 21 |  |  |  |  |  |  | sub align ($@) { | 
| 22 | 7 |  |  | 7 | 1 | 7041 | my $ali = Text::Aligner->new( shift); | 
| 23 | 7 | 100 |  |  |  | 48 | $ali->_alloc( map ref eq 'SCALAR' ? $$_ : $_, @_); | 
| 24 | 7 | 100 |  |  |  | 34 | if ( defined wantarray ) { | 
| 25 | 4 | 50 |  |  |  | 23 | my @just = map $ali->_justify( ref eq 'SCALAR' ? $$_ : $_), @_; | 
| 26 | 4 | 100 |  |  |  | 35 | return @just if wantarray; | 
| 27 | 1 |  |  |  |  | 10 | return join "\n", @just, ''; | 
| 28 |  |  |  |  |  |  | } else { | 
| 29 | 3 |  |  |  |  | 8 | for ( @_ ) { | 
| 30 | 8 | 100 |  |  |  | 24 | $_ = $ali->_justify( $_) for ref eq 'SCALAR' ? $$_ : $_; # one-shot | 
| 31 |  |  |  |  |  |  | } | 
| 32 |  |  |  |  |  |  | } | 
| 33 |  |  |  |  |  |  | } | 
| 34 |  |  |  |  |  |  |  | 
| 35 |  |  |  |  |  |  | ### class Text::Aligner | 
| 36 |  |  |  |  |  |  |  | 
| 37 |  |  |  |  |  |  | sub _new { # internal constructor | 
| 38 | 68 |  |  | 68 |  | 107 | my $class = shift; | 
| 39 | 68 |  |  |  |  | 119 | my ( $width, $pos) = @_; # both method-or-coderef (this is very general) | 
| 40 | 68 |  |  |  |  | 142 | bless { | 
| 41 |  |  |  |  |  |  | width => $width, | 
| 42 |  |  |  |  |  |  | pos => $pos, | 
| 43 |  |  |  |  |  |  | left => Text::Aligner::MaxKeeper->new, | 
| 44 |  |  |  |  |  |  | right => Text::Aligner::MaxKeeper->new, | 
| 45 |  |  |  |  |  |  | }, $class; | 
| 46 |  |  |  |  |  |  | } | 
| 47 |  |  |  |  |  |  |  | 
| 48 |  |  |  |  |  |  | # Construct an aligner | 
| 49 |  |  |  |  |  |  | sub new { | 
| 50 | 71 |  |  | 71 | 1 | 708 | my ( $class, $spec) = @_; | 
| 51 | 71 |  | 100 |  |  | 267 | $spec ||= 0; # left alignment is default | 
| 52 | 71 |  |  |  |  | 86 | my $al; | 
| 53 | 71 | 100 | 66 |  |  | 254 | if ( !ref( $spec) and $spec =~ s/^auto/num/ ) { | 
| 54 | 3 |  |  |  |  | 15 | $al = Text::Aligner::Auto->_new( $spec); | 
| 55 |  |  |  |  |  |  | } else { | 
| 56 | 68 |  |  |  |  | 118 | $al = $class->_new( _compile_alispec( $spec)); | 
| 57 |  |  |  |  |  |  | } | 
| 58 | 71 |  |  |  |  | 168 | $al; | 
| 59 |  |  |  |  |  |  | } | 
| 60 |  |  |  |  |  |  |  | 
| 61 |  |  |  |  |  |  | # return left and right field widths for an object | 
| 62 |  |  |  |  |  |  | sub _measure0 { | 
| 63 | 0 |  |  | 0 |  | 0 | my $al = shift; | 
| 64 | 0 |  |  |  |  | 0 | my $obj = shift; | 
| 65 | 0 | 0 |  |  |  | 0 | $obj = '' unless defined $obj; | 
| 66 | 0 |  |  |  |  | 0 | my ( $w, $p); | 
| 67 | 0 | 0 |  |  |  | 0 | if ( ref $obj ) { | 
| 68 | 0 |  |  |  |  | 0 | ( $w, $p) = ( $obj->$al->{ width}->(), $obj->$al->{ pos}->() ); | 
| 69 |  |  |  |  |  |  | } else { | 
| 70 | 0 |  |  |  |  | 0 | ( $w, $p) = ( $al->{ width}->( $obj), $al->{ pos}->( $obj) ); | 
| 71 |  |  |  |  |  |  | } | 
| 72 | 0 |  | 0 |  |  | 0 | $_ ||= 0 for $w, $p; | 
| 73 | 0 |  |  |  |  | 0 | ( $p, $w - $p); | 
| 74 |  |  |  |  |  |  | } | 
| 75 |  |  |  |  |  |  |  | 
| 76 | 2 |  |  | 2 |  | 1365 | use Term::ANSIColor 2.02; | 
|  | 2 |  |  |  |  | 19075 |  | 
|  | 2 |  |  |  |  | 2955 |  | 
| 77 |  |  |  |  |  |  |  | 
| 78 |  |  |  |  |  |  | # return left and right field widths for an object | 
| 79 |  |  |  |  |  |  | sub _measure { | 
| 80 | 482 |  |  | 482 |  | 761 | my $al = shift; | 
| 81 | 482 |  |  |  |  | 627 | my $obj = shift; | 
| 82 | 482 | 100 |  |  |  | 946 | $obj = '' unless defined $obj; | 
| 83 | 482 |  |  |  |  | 653 | my ( $wmeth, $pmeth) = @{ $al}{ qw( width pos)}; | 
|  | 482 |  |  |  |  | 992 |  | 
| 84 |  |  |  |  |  |  |  | 
| 85 |  |  |  |  |  |  | # support colorized strings | 
| 86 | 482 | 50 |  |  |  | 1314 | $obj = Term::ANSIColor::colorstrip($obj) unless ref $obj; | 
| 87 |  |  |  |  |  |  |  | 
| 88 | 482 | 50 |  |  |  | 4981 | my $w = ref $wmeth ? $wmeth->( $obj) : $obj->$wmeth; | 
| 89 | 482 | 50 |  |  |  | 985 | my $p = ref $pmeth ? $pmeth->( $obj) : $obj->$pmeth; | 
| 90 | 482 |  | 100 |  |  | 1866 | $_ ||= 0 for $w, $p; | 
| 91 | 482 |  |  |  |  | 1077 | ( $p, $w - $p); | 
| 92 |  |  |  |  |  |  | } | 
| 93 |  |  |  |  |  |  |  | 
| 94 |  |  |  |  |  |  | # Return left and right maxima, or nothing if the aligner is empty | 
| 95 |  |  |  |  |  |  | sub _status { | 
| 96 | 734 |  |  | 734 |  | 1335 | my @lr = ( $_[ 0]->{ left}->max, $_[ 0]->{ right}->max); | 
| 97 |  |  |  |  |  |  | # $l and $r should be both defined or undefined, unless the | 
| 98 |  |  |  |  |  |  | # MaxKeeper memory is corrupted by forgetting unremembered things. | 
| 99 | 734 | 100 | 66 |  |  | 2177 | return unless defined( $lr[ 0]) and defined( $lr[ 1]); | 
| 100 | 632 |  |  |  |  | 1453 | @lr; | 
| 101 |  |  |  |  |  |  | } | 
| 102 |  |  |  |  |  |  |  | 
| 103 |  |  |  |  |  |  | # remember alignment requirements | 
| 104 |  |  |  |  |  |  | sub _alloc { | 
| 105 | 101 |  |  | 101 |  | 2248 | my $al = shift; | 
| 106 | 101 |  |  |  |  | 178 | for ( @_ ) { | 
| 107 |  |  |  |  |  |  | #       $_ ||= ''; print "allocing '$_'\n"; | 
| 108 | 110 |  |  |  |  | 168 | my ( $l, $r) = $al->_measure( $_); | 
| 109 | 110 |  |  |  |  | 288 | $al->{ left}->remember( $l); # space needed left of pos | 
| 110 | 110 |  |  |  |  | 182 | $al->{ right}->remember( $r); # ...and right of pos | 
| 111 |  |  |  |  |  |  | } | 
| 112 | 101 |  |  |  |  | 136 | $al; | 
| 113 |  |  |  |  |  |  | } | 
| 114 |  |  |  |  |  |  |  | 
| 115 |  |  |  |  |  |  | # release alignment requirement.  it disturbs an aligner deeply to forget | 
| 116 |  |  |  |  |  |  | # things it hasn't remembered.  the effects may be delayed. | 
| 117 |  |  |  |  |  |  | sub _forget { | 
| 118 | 36 |  |  | 36 |  | 24705 | my $al = shift; | 
| 119 | 36 | 50 |  |  |  | 149 | for ( map defined() ? $_ : '', @_ ) { | 
| 120 |  |  |  |  |  |  | #       print "forgetting '$_'\n"; | 
| 121 | 30 |  |  |  |  | 72 | my ( $l, $r) = $al->_measure( $_); | 
| 122 | 30 |  |  |  |  | 99 | $al->{ left}->forget( $l); | 
| 123 | 30 |  |  |  |  | 51 | $al->{ right}->forget( $r); | 
| 124 |  |  |  |  |  |  | } | 
| 125 | 36 |  |  |  |  | 81 | $al; | 
| 126 |  |  |  |  |  |  | } | 
| 127 |  |  |  |  |  |  |  | 
| 128 |  |  |  |  |  |  | sub _spaces { | 
| 129 | 684 |  |  | 684 |  | 1027 | my ($repeat_count) = @_; | 
| 130 | 684 | 100 |  |  |  | 2250 | return (($repeat_count > 0) ? (' ' x $repeat_count) : ''); | 
| 131 |  |  |  |  |  |  | } | 
| 132 |  |  |  |  |  |  |  | 
| 133 |  |  |  |  |  |  | # justify a string.  a string is aligned within the aligner's field, and | 
| 134 |  |  |  |  |  |  | # filled with blanks or cut to size, as appropriate.  a string that has | 
| 135 |  |  |  |  |  |  | # been allocated will never be trimmed (that is the point of allocation). | 
| 136 |  |  |  |  |  |  | # if the aligner is empty it returns the string unaltered. | 
| 137 |  |  |  |  |  |  | sub _justify { | 
| 138 | 342 |  |  | 342 |  | 137941 | my $al = shift; | 
| 139 | 342 |  |  |  |  | 549 | my $str  = shift; | 
| 140 |  |  |  |  |  |  | #   print "justifying '$str'\n"; | 
| 141 | 342 |  |  |  |  | 523 | $str .= ''; # stringify (objects, numbers, undef) | 
| 142 | 342 |  |  |  |  | 678 | my ( $l_pad, $r_pad) = $al->_padding( $str); | 
| 143 | 342 | 100 |  |  |  | 712 | substr( $str, 0, -$l_pad) = '' if $l_pad < 0; # trim if negative | 
| 144 | 342 | 100 |  |  |  | 602 | substr( $str, $r_pad) = '' if $r_pad < 0; # ... both ends | 
| 145 | 342 |  |  |  |  | 574 | return _spaces($l_pad) . $str . _spaces($r_pad); # pad if positive | 
| 146 |  |  |  |  |  |  | } | 
| 147 |  |  |  |  |  |  |  | 
| 148 |  |  |  |  |  |  | # return two numbers that indicate how many blanks are needed on each side | 
| 149 |  |  |  |  |  |  | # of a string to justify it.  Negative values mean trim that many characters. | 
| 150 |  |  |  |  |  |  | # an empty aligner returns ( 0, 0), so doesn't change anything. | 
| 151 |  |  |  |  |  |  | sub _padding { | 
| 152 | 342 |  |  | 342 |  | 472 | my $al = shift; | 
| 153 | 342 |  |  |  |  | 448 | my $str = shift; | 
| 154 | 342 |  |  |  |  | 594 | my ( $this_l, $this_r) = $al->_measure( $str); | 
| 155 | 342 |  |  |  |  | 597 | my ( $l_pad, $r_pad) = ( 0, 0); | 
| 156 | 342 | 100 |  |  |  | 613 | if ( $al->_status ) { | 
| 157 | 288 |  |  |  |  | 471 | ( $l_pad, $r_pad) = $al->_status; | 
| 158 | 288 |  |  |  |  | 422 | $l_pad -= $this_l; | 
| 159 | 288 |  |  |  |  | 364 | $r_pad -= $this_r; | 
| 160 |  |  |  |  |  |  | } | 
| 161 | 342 |  |  |  |  | 719 | ( $l_pad, $r_pad); | 
| 162 |  |  |  |  |  |  | } | 
| 163 |  |  |  |  |  |  |  | 
| 164 |  |  |  |  |  |  | # _compile_alispec() returns positioners according to specification.  In | 
| 165 |  |  |  |  |  |  | # effect, it is the interpreter for alignment specifications. | 
| 166 |  |  |  |  |  |  |  | 
| 167 |  |  |  |  |  |  | sub _compile_alispec { # it's a dirty job... | 
| 168 | 482 |  |  | 482 |  | 836 | my $width = sub { length shift }; # this is always so for string aligners | 
|  | 82 |  |  | 82 |  | 7246 |  | 
| 169 | 82 |  |  |  |  | 127 | my $pos; # the positioner we actually compile | 
| 170 | 82 |  | 100 |  |  | 247 | local $_ = shift || ''; # alignment specification | 
| 171 | 82 | 100 |  |  |  | 163 | if ( ref() eq 'Regexp' ) { | 
| 172 | 2 |  |  |  |  | 6 | my $regex = $_; # lexical copy! | 
| 173 |  |  |  |  |  |  | $pos = sub { | 
| 174 | 24 |  |  | 24 |  | 11635 | local $_ = shift; | 
| 175 | 24 | 100 |  |  |  | 164 | return m/$regex/ ? $-[ 0] : length; # assume match after string | 
| 176 | 2 |  |  |  |  | 6 | }; | 
| 177 |  |  |  |  |  |  | } else { | 
| 178 | 80 |  |  |  |  | 136 | s/^left/0/; | 
| 179 | 80 |  |  |  |  | 108 | s/^center/0.5/; | 
| 180 | 80 |  |  |  |  | 122 | s/^right/1/; | 
| 181 | 80 | 100 |  |  |  | 139 | if ( _is_number( $_) ) { | 
|  |  | 100 |  |  |  |  |  | 
| 182 | 11 |  |  |  |  | 25 | my $proportion = $_; # use lexical copy | 
| 183 | 11 |  |  | 204 |  | 43 | $pos = sub { int( $proportion*length shift) }; | 
|  | 204 |  |  |  |  | 17416 |  | 
| 184 |  |  |  |  |  |  | } elsif ( $_ =~ /^(?:num|point)(?:\((.*))?/ ) { | 
| 185 | 8 | 100 |  |  |  | 33 | my $point = defined $1 ? $1 : ''; | 
| 186 | 8 |  |  |  |  | 19 | $point =~ s/\)$//; # ignore trailing paren, if present | 
| 187 | 8 | 100 |  |  |  | 28 | length $point or $point = '.'; | 
| 188 | 141 |  |  | 141 |  | 12269 | $pos = sub { index( shift() . $point, $point) } | 
| 189 | 8 |  |  |  |  | 29 | } else { | 
| 190 | 61 |  |  | 197 |  | 157 | $pos = sub { 0 }; | 
|  | 197 |  |  |  |  | 283 |  | 
| 191 |  |  |  |  |  |  | } | 
| 192 |  |  |  |  |  |  | } | 
| 193 | 82 |  |  |  |  | 242 | ( $width, $pos); | 
| 194 |  |  |  |  |  |  | } | 
| 195 |  |  |  |  |  |  |  | 
| 196 |  |  |  |  |  |  | # decide if a string is a number. (see perlfaq4). | 
| 197 |  |  |  |  |  |  | sub _is_number { | 
| 198 | 179 |  |  | 179 |  | 3555 | my ($x) = @_; | 
| 199 | 179 | 100 |  |  |  | 354 | return 0 unless defined $x; | 
| 200 | 170 | 100 |  |  |  | 639 | return 0 if $x !~ /\d/; | 
| 201 | 65 | 100 |  |  |  | 371 | return 1 if $x =~ /^-?\d+\.?\d*$/; | 
| 202 | 14 |  |  |  |  | 30 | $x = Term::ANSIColor::colorstrip($x); | 
| 203 | 14 |  |  |  |  | 188 | $x =~ /^-?\d+\.?\d*$/ | 
| 204 |  |  |  |  |  |  | } | 
| 205 |  |  |  |  |  |  |  | 
| 206 |  |  |  |  |  |  | package Text::Aligner::Auto; | 
| 207 |  |  |  |  |  |  | # Combined numeric and left alignment.  Numbers are aligned numerically, | 
| 208 |  |  |  |  |  |  | # other strings are left-aligned.  The resulting columns are interleaved | 
| 209 |  |  |  |  |  |  | # flush left and filled on the right if necessary. | 
| 210 |  |  |  |  |  |  |  | 
| 211 |  |  |  |  |  |  | sub _new { # only called by Text::Aligner->new() | 
| 212 | 3 |  |  | 3 |  | 5 | my $class = shift; | 
| 213 | 3 |  |  |  |  | 5 | my $numspec = shift; # currently ignored | 
| 214 | 3 |  |  |  |  | 11 | bless { | 
| 215 |  |  |  |  |  |  | num => Text::Aligner->new( 'num'),    # align numbers among themselves | 
| 216 |  |  |  |  |  |  | other => Text::Aligner->new,          # left-align anything else | 
| 217 |  |  |  |  |  |  | }, $class; | 
| 218 |  |  |  |  |  |  | } | 
| 219 |  |  |  |  |  |  |  | 
| 220 |  |  |  |  |  |  | sub _alloc { | 
| 221 | 8 |  |  | 8 |  | 537 | my $aa = shift; | 
| 222 | 8 |  |  |  |  | 21 | my @num = grep _is_number( $_), @_; | 
| 223 | 8 |  |  |  |  | 19 | my @other = grep !_is_number( $_), @_; | 
| 224 | 8 |  |  |  |  | 23 | $aa->{ num}->_alloc( @num); | 
| 225 | 8 |  |  |  |  | 21 | $aa->{ other}->_alloc( @other); | 
| 226 | 8 |  |  |  |  | 16 | $aa; | 
| 227 |  |  |  |  |  |  | } | 
| 228 |  |  |  |  |  |  |  | 
| 229 |  |  |  |  |  |  | sub _forget { | 
| 230 | 6 |  |  | 6 |  | 6273 | my $aa = shift; | 
| 231 | 6 |  |  |  |  | 20 | $aa->{ num}->_forget( grep _is_number( $_), @_); | 
| 232 | 6 |  |  |  |  | 15 | $aa->{ other}->_forget( grep !_is_number( $_), @_); | 
| 233 | 6 |  |  |  |  | 13 | $aa; | 
| 234 |  |  |  |  |  |  | } | 
| 235 |  |  |  |  |  |  |  | 
| 236 |  |  |  |  |  |  | # Justify as required | 
| 237 |  |  |  |  |  |  | sub _justify { | 
| 238 | 52 |  |  | 52 |  | 33996 | my ( $aa, $str) = @_; | 
| 239 |  |  |  |  |  |  | # align according to type | 
| 240 | 52 | 100 |  |  |  | 122 | $str = $aa->{ _is_number( $str) ? 'num' : 'other'}->_justify( $str); | 
| 241 | 52 |  |  |  |  | 136 | my $combi = Text::Aligner->new; # left-justify pre-aligned string | 
| 242 |  |  |  |  |  |  | # initialise to size of partial aligners.  (don't initialise from | 
| 243 |  |  |  |  |  |  | # empty aligner) | 
| 244 | 52 | 100 |  |  |  | 115 | $combi->_alloc( $aa->{ num}->_justify( '')) if $aa->{ num}->_status; | 
| 245 | 52 | 100 |  |  |  | 101 | $combi->_alloc( $aa->{ other}->_justify( '')) if $aa->{ other}->_status; | 
| 246 | 52 |  |  |  |  | 105 | $combi->_justify( $str); | 
| 247 |  |  |  |  |  |  | } | 
| 248 |  |  |  |  |  |  |  | 
| 249 |  |  |  |  |  |  | # for convenience | 
| 250 |  |  |  |  |  |  | BEGIN { # import _is_number() | 
| 251 | 2 |  |  | 2 |  | 645 | *_is_number = \ &Text::Aligner::_is_number; | 
| 252 |  |  |  |  |  |  | } | 
| 253 |  |  |  |  |  |  |  | 
| 254 |  |  |  |  |  |  | package Text::Aligner::MaxKeeper; | 
| 255 |  |  |  |  |  |  | # Keep the maximum of a dynamic set of numbers.  Optimized for the case of | 
| 256 |  |  |  |  |  |  | # a relatively small range of numbers that may occur repeatedly. | 
| 257 |  |  |  |  |  |  |  | 
| 258 |  |  |  |  |  |  | sub new { | 
| 259 | 137 |  |  | 137 |  | 644 | bless { | 
| 260 |  |  |  |  |  |  | max => undef, | 
| 261 |  |  |  |  |  |  | seen => {}, | 
| 262 |  |  |  |  |  |  | }, shift; | 
| 263 |  |  |  |  |  |  | } | 
| 264 |  |  |  |  |  |  |  | 
| 265 | 1475 |  |  | 1475 |  | 2493 | sub max { $_[ 0]->{ max} } | 
| 266 |  |  |  |  |  |  |  | 
| 267 |  |  |  |  |  |  | sub remember { | 
| 268 | 228 |  |  | 228 |  | 370 | my ( $mk, $val) = @_; | 
| 269 | 228 |  |  |  |  | 473 | _to_max( $mk->{ max}, $val); | 
| 270 | 228 |  |  |  |  | 482 | $mk->{ seen}->{ $val}++; | 
| 271 | 228 |  |  |  |  | 341 | $mk; | 
| 272 |  |  |  |  |  |  | } | 
| 273 |  |  |  |  |  |  |  | 
| 274 |  |  |  |  |  |  | sub forget { | 
| 275 | 65 |  |  | 65 |  | 119 | my ( $mk, $val) = @_; | 
| 276 | 65 | 50 |  |  |  | 144 | if ( exists $mk->{ seen}->{ $val} ) { | 
| 277 | 65 |  |  |  |  | 88 | my $seen = $mk->{ seen}; | 
| 278 | 65 | 100 |  |  |  | 123 | unless ( --$seen->{ $val} ) { | 
| 279 | 63 |  |  |  |  | 111 | delete $seen->{ $val}; | 
| 280 | 63 | 100 |  |  |  | 119 | if ( $mk->{ max} == $val ) { | 
| 281 |  |  |  |  |  |  | # lost the maximum, recalculate | 
| 282 | 62 |  |  |  |  | 93 | undef $mk->{ max}; | 
| 283 | 62 |  |  |  |  | 140 | _to_max( $mk->{ max}, keys %$seen); | 
| 284 |  |  |  |  |  |  | } | 
| 285 |  |  |  |  |  |  | } | 
| 286 |  |  |  |  |  |  | } | 
| 287 | 65 |  |  |  |  | 99 | $mk; | 
| 288 |  |  |  |  |  |  | } | 
| 289 |  |  |  |  |  |  |  | 
| 290 |  |  |  |  |  |  | sub _to_max { | 
| 291 | 290 |  |  | 290 |  | 407 | my $var = \ shift; | 
| 292 | 290 |  | 100 |  |  | 1054 | defined $_ and ( not defined $$var or $$var < $_) and $$var = $_ for @_; | 
|  |  |  | 66 |  |  |  |  | 
|  |  |  | 66 |  |  |  |  | 
| 293 | 290 |  |  |  |  | 421 | $$var; | 
| 294 |  |  |  |  |  |  | } | 
| 295 |  |  |  |  |  |  |  | 
| 296 |  |  |  |  |  |  | ########################################### main pod documentation begin ## | 
| 297 |  |  |  |  |  |  |  | 
| 298 |  |  |  |  |  |  |  | 
| 299 |  |  |  |  |  |  | 1; #this line is important and will help the module return a true value | 
| 300 |  |  |  |  |  |  |  | 
| 301 |  |  |  |  |  |  | __END__ |