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__ |