line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
#!/usr/bin/perl |
2
|
|
|
|
|
|
|
|
3
|
|
|
|
|
|
|
package main; |
4
|
|
|
|
|
|
|
|
5
|
|
|
|
|
|
|
our $options; |
6
|
|
|
|
|
|
|
our $config; |
7
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
package ChordPro::Output::ChordPro; |
9
|
|
|
|
|
|
|
|
10
|
5
|
|
|
5
|
|
1024
|
use strict; |
|
5
|
|
|
|
|
11
|
|
|
5
|
|
|
|
|
207
|
|
11
|
5
|
|
|
5
|
|
30
|
use warnings; |
|
5
|
|
|
|
|
12
|
|
|
5
|
|
|
|
|
188
|
|
12
|
|
|
|
|
|
|
|
13
|
5
|
|
|
5
|
|
134
|
use ChordPro::Output::Common; |
|
5
|
|
|
|
|
11
|
|
|
5
|
|
|
|
|
387
|
|
14
|
5
|
|
|
5
|
|
45
|
use ChordPro::Utils qw( fq qquote demarkup is_true is_ttrue ); |
|
5
|
|
|
|
|
17
|
|
|
5
|
|
|
|
|
20489
|
|
15
|
|
|
|
|
|
|
|
16
|
|
|
|
|
|
|
my $re_meta; |
17
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
sub generate_songbook { |
19
|
23
|
|
|
23
|
0
|
84
|
my ( $self, $sb ) = @_; |
20
|
|
|
|
|
|
|
|
21
|
|
|
|
|
|
|
# Skip empty songbooks. |
22
|
23
|
50
|
|
|
|
71
|
return [] unless eval { $sb->{songs}->[0]->{body} }; |
|
23
|
|
|
|
|
127
|
|
23
|
|
|
|
|
|
|
|
24
|
|
|
|
|
|
|
# Build regex for the known metadata items. |
25
|
|
|
|
|
|
|
$re_meta = join( '|', |
26
|
23
|
|
|
|
|
153
|
map { quotemeta } |
|
322
|
|
|
|
|
828
|
|
27
|
|
|
|
|
|
|
"title", "subtitle", |
28
|
|
|
|
|
|
|
"artist", "composer", "lyricist", "arranger", |
29
|
|
|
|
|
|
|
"album", "copyright", "year", |
30
|
|
|
|
|
|
|
"key", "time", "tempo", "capo", "duration" ); |
31
|
23
|
|
|
|
|
434
|
$re_meta = qr/^($re_meta)$/; |
32
|
|
|
|
|
|
|
|
33
|
23
|
|
|
|
|
75
|
my @book; |
34
|
|
|
|
|
|
|
|
35
|
23
|
|
|
|
|
54
|
foreach my $song ( @{$sb->{songs}} ) { |
|
23
|
|
|
|
|
94
|
|
36
|
26
|
100
|
|
|
|
83
|
if ( @book ) { |
37
|
3
|
50
|
|
|
|
11
|
push(@book, "") if $options->{'backend-option'}->{tidy}; |
38
|
3
|
|
|
|
|
8
|
push(@book, "{new_song}"); |
39
|
|
|
|
|
|
|
} |
40
|
26
|
|
|
|
|
72
|
push(@book, @{generate_song($song)}); |
|
26
|
|
|
|
|
113
|
|
41
|
|
|
|
|
|
|
} |
42
|
|
|
|
|
|
|
|
43
|
23
|
|
|
|
|
98
|
push( @book, ""); |
44
|
23
|
|
|
|
|
108
|
\@book; |
45
|
|
|
|
|
|
|
} |
46
|
|
|
|
|
|
|
|
47
|
|
|
|
|
|
|
my $lyrics_only = 0; |
48
|
|
|
|
|
|
|
my $variant = 'cho'; |
49
|
|
|
|
|
|
|
my $rechorus; |
50
|
|
|
|
|
|
|
|
51
|
|
|
|
|
|
|
sub upd_config { |
52
|
26
|
|
|
26
|
0
|
117
|
$rechorus = $::config->{chordpro}->{chorus}->{recall}; |
53
|
26
|
|
|
|
|
155
|
$lyrics_only = 2 * $::config->{settings}->{'lyrics-only'}; |
54
|
|
|
|
|
|
|
} |
55
|
|
|
|
|
|
|
|
56
|
|
|
|
|
|
|
sub generate_song { |
57
|
26
|
|
|
26
|
0
|
71
|
my ( $s ) = @_; |
58
|
|
|
|
|
|
|
|
59
|
26
|
|
|
|
|
115
|
my $tidy = $options->{'backend-option'}->{tidy}; |
60
|
26
|
|
50
|
|
|
164
|
my $structured = ( $options->{'backend-option'}->{structure} // '' ) eq 'structured'; |
61
|
|
|
|
|
|
|
# $s->structurize if ++$structured; |
62
|
26
|
|
50
|
|
|
157
|
$variant = $options->{'backend-option'}->{variant} || 'cho'; |
63
|
26
|
|
|
|
|
60
|
my $seq = $options->{'backend-option'}->{seq}; |
64
|
26
|
|
|
|
|
76
|
my $expand = $options->{'backend-option'}->{expand}; |
65
|
26
|
|
|
|
|
76
|
my $msp = $variant eq "msp"; |
66
|
26
|
|
|
|
|
98
|
upd_config(); |
67
|
|
|
|
|
|
|
|
68
|
26
|
|
|
|
|
163
|
my @s; |
69
|
|
|
|
|
|
|
my %imgs; |
70
|
|
|
|
|
|
|
|
71
|
26
|
50
|
|
|
|
125
|
if ( $s->{preamble} ) { |
72
|
0
|
|
|
|
|
0
|
@s = @{ $s->{preamble} }; |
|
0
|
|
|
|
|
0
|
|
73
|
|
|
|
|
|
|
} |
74
|
|
|
|
|
|
|
|
75
|
|
|
|
|
|
|
push(@s, "{title: " . fq($s->{meta}->{title}->[0]) . "}") |
76
|
26
|
50
|
|
|
|
205
|
if defined $s->{meta}->{title}; |
77
|
26
|
100
|
|
|
|
107
|
if ( defined $s->{subtitle} ) { |
78
|
6
|
|
|
|
|
16
|
push(@s, map { +"{subtitle: ".fq($_)."}" } @{$s->{subtitle}}); |
|
6
|
|
|
|
|
17
|
|
|
6
|
|
|
|
|
20
|
|
79
|
|
|
|
|
|
|
} |
80
|
|
|
|
|
|
|
|
81
|
26
|
50
|
|
|
|
107
|
if ( $s->{meta} ) { |
82
|
26
|
50
|
|
|
|
74
|
if ( $msp ) { |
83
|
0
|
|
0
|
|
|
0
|
$s->{meta}->{source} //= [ "Lead Sheet" ]; |
84
|
0
|
0
|
0
|
|
|
0
|
$s->{meta}->{custom2} //= [ $seq ] if defined $seq; |
85
|
|
|
|
|
|
|
} |
86
|
|
|
|
|
|
|
# Known ones 'as is'. |
87
|
26
|
|
|
|
|
49
|
my %used; |
88
|
26
|
|
|
|
|
57
|
foreach my $k ( sort keys %{ $s->{meta} } ) { |
|
26
|
|
|
|
|
260
|
|
89
|
130
|
100
|
|
|
|
435
|
next if $k =~ /^(?:title|subtitle)$/; |
90
|
98
|
100
|
|
|
|
475
|
if ( $k =~ $re_meta ) { |
91
|
11
|
|
|
|
|
23
|
push( @s, map { +"{$k: ".fq($_)."}" } @{ $s->{meta}->{$k} } ); |
|
12
|
|
|
|
|
46
|
|
|
11
|
|
|
|
|
39
|
|
92
|
11
|
|
|
|
|
38
|
$used{$k}++; |
93
|
|
|
|
|
|
|
} |
94
|
|
|
|
|
|
|
} |
95
|
|
|
|
|
|
|
# Unknowns with meta prefix. |
96
|
26
|
|
|
|
|
94
|
foreach my $k ( sort keys %{ $s->{meta} } ) { |
|
26
|
|
|
|
|
163
|
|
97
|
130
|
100
|
|
|
|
286
|
next if $used{$k}; |
98
|
119
|
50
|
|
|
|
500
|
next if $k =~ /^(?:title|subtitle|songindex|key_.*|chords|numchords)$/; |
99
|
0
|
0
|
|
|
|
0
|
next if $k =~ /^_/; |
100
|
0
|
|
|
|
|
0
|
push( @s, map { +"{meta: $k ".fq($_)."}" } @{ $s->{meta}->{$k} } ); |
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
101
|
|
|
|
|
|
|
} |
102
|
|
|
|
|
|
|
} |
103
|
|
|
|
|
|
|
|
104
|
26
|
50
|
|
|
|
128
|
if ( $s->{settings} ) { |
105
|
26
|
|
|
|
|
75
|
foreach ( sort keys %{ $s->{settings} } ) { |
|
26
|
|
|
|
|
121
|
|
106
|
16
|
100
|
|
|
|
64
|
if ( $_ eq "diagrams" ) { |
|
|
100
|
|
|
|
|
|
107
|
3
|
100
|
|
|
|
20
|
next if $s->{settings}->{diagrampos}; |
108
|
2
|
|
|
|
|
7
|
my $v = $s->{settings}->{$_}; |
109
|
2
|
50
|
|
|
|
10
|
if ( is_ttrue($v) ) { |
|
|
50
|
|
|
|
|
|
110
|
0
|
|
|
|
|
0
|
$v = "on"; |
111
|
|
|
|
|
|
|
} |
112
|
|
|
|
|
|
|
elsif ( is_true($v) ) { |
113
|
|
|
|
|
|
|
} |
114
|
|
|
|
|
|
|
else { |
115
|
2
|
|
|
|
|
5
|
$v = "off"; |
116
|
|
|
|
|
|
|
} |
117
|
2
|
|
|
|
|
13
|
push(@s, "{diagrams: $v}"); |
118
|
|
|
|
|
|
|
} |
119
|
|
|
|
|
|
|
elsif ( $_ eq "diagrampos" ) { |
120
|
1
|
|
|
|
|
5
|
my $v = $s->{settings}->{$_}; |
121
|
1
|
|
|
|
|
6
|
push(@s, "{diagrams: $v}"); |
122
|
|
|
|
|
|
|
} |
123
|
|
|
|
|
|
|
else { |
124
|
12
|
|
|
|
|
53
|
push(@s, "{$_: " . $s->{settings}->{$_} . "}"); |
125
|
|
|
|
|
|
|
} |
126
|
|
|
|
|
|
|
} |
127
|
|
|
|
|
|
|
} |
128
|
|
|
|
|
|
|
|
129
|
26
|
50
|
|
|
|
77
|
push(@s, "") if $tidy; |
130
|
|
|
|
|
|
|
|
131
|
|
|
|
|
|
|
# Move a trailing list of chords to the beginning, so the chords |
132
|
|
|
|
|
|
|
# are defined when the song is parsed. |
133
|
26
|
50
|
33
|
|
|
52
|
if ( @{ $s->{body} } && $s->{body}->[-1]->{type} eq "diagrams" |
|
26
|
|
33
|
|
|
264
|
|
134
|
|
|
|
|
|
|
&& $s->{body}->[-1]->{origin} ne "__CLI__" |
135
|
|
|
|
|
|
|
) { |
136
|
0
|
|
|
|
|
0
|
unshift( @{ $s->{body} }, pop( @{ $s->{body} } ) ); |
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
137
|
|
|
|
|
|
|
} |
138
|
|
|
|
|
|
|
|
139
|
26
|
100
|
|
|
|
98
|
if ( $s->{define} ) { |
140
|
9
|
|
|
|
|
29
|
foreach my $info ( @{ $s->{define} } ) { |
|
9
|
|
|
|
|
44
|
|
141
|
27
|
|
|
|
|
73
|
my $t = "{define: " . $info->{name}; |
142
|
27
|
50
|
|
|
|
81
|
if ( $info->{copyall} ) { |
|
|
50
|
|
|
|
|
|
143
|
0
|
|
|
|
|
0
|
$t .= " copyall " . $info->{copyall}; |
144
|
|
|
|
|
|
|
} |
145
|
|
|
|
|
|
|
elsif ( $info->{copy} ) { |
146
|
0
|
|
|
|
|
0
|
$t .= " copy " . $info->{copy}; |
147
|
|
|
|
|
|
|
} |
148
|
27
|
|
|
|
|
62
|
for ( qw( display ) ) { |
149
|
27
|
100
|
|
|
|
92
|
next unless defined $info->{$_}; |
150
|
2
|
|
|
|
|
18
|
$t .= " $_ " . qquote($info->{$_}->name ); |
151
|
|
|
|
|
|
|
} |
152
|
27
|
|
|
|
|
67
|
for ( qw( format ) ) { |
153
|
27
|
100
|
|
|
|
77
|
next unless defined $info->{$_}; |
154
|
9
|
|
|
|
|
43
|
$t .= " $_ " . qquote($info->{$_} ); |
155
|
|
|
|
|
|
|
} |
156
|
27
|
|
|
|
|
74
|
$t .= " base-fret " . $info->{base}; |
157
|
|
|
|
|
|
|
$t .= " frets " . |
158
|
162
|
100
|
|
|
|
400
|
join(" ", map { $_ < 0 ? "N" : $_ } @{$info->{frets}}) |
|
27
|
|
|
|
|
59
|
|
159
|
27
|
50
|
|
|
|
89
|
if $info->{frets}; |
160
|
|
|
|
|
|
|
$t .= " fingers " . |
161
|
30
|
50
|
|
|
|
71
|
join(" ", map { $_ < 0 ? "N" : $_ } @{$info->{fingers}}) |
|
5
|
|
|
|
|
11
|
|
162
|
27
|
100
|
100
|
|
|
111
|
if $info->{fingers} && @{$info->{fingers}}; |
|
15
|
|
|
|
|
63
|
|
163
|
|
|
|
|
|
|
$t .= " keys " . |
164
|
0
|
|
|
|
|
0
|
join(" ", @{$info->{keys}}) |
165
|
27
|
50
|
66
|
|
|
90
|
if $info->{keys} && @{$info->{keys}}; |
|
11
|
|
|
|
|
36
|
|
166
|
27
|
|
|
|
|
64
|
for ( qw( diagram ) ) { |
167
|
27
|
100
|
|
|
|
79
|
next unless defined $info->{$_}; |
168
|
3
|
|
|
|
|
9
|
my $v = $info->{$_}; |
169
|
3
|
50
|
|
|
|
15
|
if ( is_true($v) ) { |
170
|
0
|
0
|
|
|
|
0
|
if ( is_ttrue($v) ) { |
171
|
0
|
|
|
|
|
0
|
next; |
172
|
|
|
|
|
|
|
} |
173
|
|
|
|
|
|
|
} |
174
|
|
|
|
|
|
|
else { |
175
|
3
|
|
|
|
|
14
|
$v = "off"; |
176
|
|
|
|
|
|
|
} |
177
|
3
|
|
|
|
|
13
|
$t .= " $_ $v"; |
178
|
|
|
|
|
|
|
} |
179
|
27
|
|
|
|
|
128
|
push(@s, $t . "}"); |
180
|
|
|
|
|
|
|
} |
181
|
9
|
50
|
|
|
|
59
|
push(@s, "") if $tidy; |
182
|
|
|
|
|
|
|
} |
183
|
|
|
|
|
|
|
|
184
|
26
|
|
|
|
|
54
|
my $ctx = ""; |
185
|
26
|
|
|
|
|
68
|
my $dumphdr = 1; |
186
|
|
|
|
|
|
|
|
187
|
26
|
100
|
66
|
|
|
166
|
if ( $s->{chords} && $variant ne 'msp' ) { |
188
|
24
|
50
|
|
|
|
137
|
$dumphdr = 0 unless $s->{chords}->{origin} eq "__CLI__"; |
189
|
|
|
|
|
|
|
push( @s, |
190
|
24
|
|
|
|
|
65
|
@{ ChordPro::Chords::list_chords |
191
|
|
|
|
|
|
|
( $s->{chords}->{chords}, |
192
|
|
|
|
|
|
|
$s->{chords}->{origin}, |
193
|
24
|
|
|
|
|
141
|
$dumphdr ) } ); |
194
|
24
|
|
|
|
|
69
|
$dumphdr = 0; |
195
|
|
|
|
|
|
|
} |
196
|
|
|
|
|
|
|
|
197
|
26
|
|
|
|
|
52
|
my @elts = @{$s->{body}}; |
|
26
|
|
|
|
|
136
|
|
198
|
26
|
|
|
|
|
120
|
while ( @elts ) { |
199
|
310
|
|
|
|
|
509
|
my $elt = shift(@elts); |
200
|
|
|
|
|
|
|
|
201
|
310
|
100
|
|
|
|
753
|
if ( $elt->{context} ne $ctx ) { |
202
|
30
|
100
|
|
|
|
126
|
push(@s, "{end_of_$ctx}") if $ctx; |
203
|
30
|
|
|
|
|
64
|
$ctx = $elt->{context}; |
204
|
30
|
100
|
|
|
|
76
|
if ( $ctx ) { |
205
|
|
|
|
|
|
|
|
206
|
18
|
|
|
|
|
51
|
my $t = "{start_of_$ctx"; |
207
|
|
|
|
|
|
|
|
208
|
18
|
100
|
|
|
|
55
|
if ( $elt->{type} eq "set" ) { |
209
|
4
|
100
|
|
|
|
15
|
if ( $elt->{name} eq "gridparams" ) { |
|
|
50
|
|
|
|
|
|
210
|
2
|
|
|
|
|
6
|
my @gridparams = @{ $elt->{value} }; |
|
2
|
|
|
|
|
10
|
|
211
|
2
|
|
|
|
|
7
|
$t .= ": "; |
212
|
2
|
50
|
|
|
|
6
|
$t .= $gridparams[2] . "+" if $gridparams[2]; |
213
|
2
|
|
|
|
|
5
|
$t .= $gridparams[0]; |
214
|
2
|
50
|
|
|
|
8
|
$t .= "x" . $gridparams[1] if $gridparams[1]; |
215
|
2
|
50
|
|
|
|
7
|
$t .= "+" . $gridparams[3] if $gridparams[3]; |
216
|
2
|
50
|
|
|
|
14
|
if ( $gridparams[4] ) { |
217
|
0
|
|
|
|
|
0
|
my $tag = $gridparams[4]; |
218
|
0
|
0
|
|
|
|
0
|
$t .= " " . $tag if $tag ne ""; |
219
|
|
|
|
|
|
|
} |
220
|
|
|
|
|
|
|
} |
221
|
|
|
|
|
|
|
elsif ( $elt->{name} eq "label" ) { |
222
|
2
|
|
|
|
|
6
|
my $tag = $elt->{value}; |
223
|
2
|
50
|
|
|
|
9
|
$t .= ": " . $tag if $tag ne ""; |
224
|
|
|
|
|
|
|
} |
225
|
|
|
|
|
|
|
|
226
|
|
|
|
|
|
|
} |
227
|
18
|
|
|
|
|
42
|
$t .= "}"; |
228
|
18
|
|
|
|
|
48
|
push( @s, $t ); |
229
|
|
|
|
|
|
|
} |
230
|
|
|
|
|
|
|
} |
231
|
|
|
|
|
|
|
|
232
|
310
|
100
|
|
|
|
664
|
if ( $elt->{type} eq "empty" ) { |
233
|
61
|
50
|
|
|
|
140
|
push(@s, "***SHOULD NOT HAPPEN***"), next |
234
|
|
|
|
|
|
|
if $structured; |
235
|
61
|
|
|
|
|
121
|
push( @s, "" ); |
236
|
61
|
|
|
|
|
131
|
next; |
237
|
|
|
|
|
|
|
} |
238
|
|
|
|
|
|
|
|
239
|
249
|
100
|
|
|
|
515
|
if ( $elt->{type} eq "colb" ) { |
240
|
3
|
50
|
|
|
|
17
|
next if $msp; |
241
|
3
|
|
|
|
|
11
|
push(@s, "{column_break}"); |
242
|
3
|
|
|
|
|
7
|
next; |
243
|
|
|
|
|
|
|
} |
244
|
|
|
|
|
|
|
|
245
|
246
|
100
|
|
|
|
492
|
if ( $elt->{type} eq "newpage" ) { |
246
|
3
|
50
|
|
|
|
24
|
next if $msp; |
247
|
3
|
|
|
|
|
11
|
push(@s, "{new_page}"); |
248
|
3
|
|
|
|
|
11
|
next; |
249
|
|
|
|
|
|
|
} |
250
|
|
|
|
|
|
|
|
251
|
243
|
100
|
|
|
|
498
|
if ( $elt->{type} eq "songline" ) { |
252
|
119
|
|
|
|
|
296
|
push(@s, songline( $s, $elt )); |
253
|
119
|
|
|
|
|
340
|
next; |
254
|
|
|
|
|
|
|
} |
255
|
|
|
|
|
|
|
|
256
|
124
|
100
|
|
|
|
280
|
if ( $elt->{type} eq "tabline" ) { |
257
|
24
|
|
|
|
|
45
|
push(@s, $elt->{text} ); |
258
|
24
|
|
|
|
|
48
|
next; |
259
|
|
|
|
|
|
|
} |
260
|
|
|
|
|
|
|
|
261
|
100
|
100
|
|
|
|
224
|
if ( $elt->{type} eq "gridline" ) { |
262
|
4
|
|
|
|
|
11
|
push(@s, gridline( $s, $elt )); |
263
|
4
|
|
|
|
|
12
|
next; |
264
|
|
|
|
|
|
|
} |
265
|
|
|
|
|
|
|
|
266
|
96
|
50
|
|
|
|
227
|
if ( $elt->{type} eq "verse" ) { |
267
|
0
|
0
|
|
|
|
0
|
push(@s, "") if $tidy; |
268
|
0
|
|
|
|
|
0
|
foreach my $e ( @{$elt->{body}} ) { |
|
0
|
|
|
|
|
0
|
|
269
|
0
|
0
|
|
|
|
0
|
if ( $e->{type} eq "empty" ) { |
270
|
0
|
0
|
|
|
|
0
|
push(@s, "***SHOULD NOT HAPPEN***"), next |
271
|
|
|
|
|
|
|
if $structured; |
272
|
|
|
|
|
|
|
} |
273
|
0
|
0
|
|
|
|
0
|
if ( $e->{type} eq "song" ) { |
274
|
0
|
|
|
|
|
0
|
push(@s, songline( $s, $e )); |
275
|
0
|
|
|
|
|
0
|
next; |
276
|
|
|
|
|
|
|
} |
277
|
|
|
|
|
|
|
} |
278
|
0
|
0
|
|
|
|
0
|
push(@s, "") if $tidy; |
279
|
0
|
|
|
|
|
0
|
next; |
280
|
|
|
|
|
|
|
} |
281
|
|
|
|
|
|
|
|
282
|
96
|
50
|
|
|
|
197
|
if ( $elt->{type} eq "chorus" ) { |
283
|
0
|
0
|
|
|
|
0
|
push(@s, "") if $tidy; |
284
|
0
|
|
|
|
|
0
|
push(@s, "{start_of_chorus*}"); |
285
|
0
|
|
|
|
|
0
|
foreach my $e ( @{$elt->{body}} ) { |
|
0
|
|
|
|
|
0
|
|
286
|
0
|
0
|
|
|
|
0
|
if ( $e->{type} eq "empty" ) { |
287
|
0
|
|
|
|
|
0
|
push(@s, ""); |
288
|
0
|
|
|
|
|
0
|
next; |
289
|
|
|
|
|
|
|
} |
290
|
0
|
0
|
|
|
|
0
|
if ( $e->{type} eq "songline" ) { |
291
|
0
|
|
|
|
|
0
|
push(@s, songline( $s, $e )); |
292
|
0
|
|
|
|
|
0
|
next; |
293
|
|
|
|
|
|
|
} |
294
|
|
|
|
|
|
|
} |
295
|
0
|
|
|
|
|
0
|
push(@s, "{end_of_chorus*}"); |
296
|
0
|
0
|
|
|
|
0
|
push(@s, "") if $tidy; |
297
|
0
|
|
|
|
|
0
|
next; |
298
|
|
|
|
|
|
|
} |
299
|
|
|
|
|
|
|
|
300
|
96
|
50
|
|
|
|
203
|
if ( $elt->{type} eq "rechorus" ) { |
301
|
0
|
0
|
0
|
|
|
0
|
if ( $msp ) { |
|
|
0
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
302
|
0
|
|
|
|
|
0
|
push( @s, "{chorus}" ); |
303
|
|
|
|
|
|
|
} |
304
|
|
|
|
|
|
|
elsif ( $rechorus->{quote} ) { |
305
|
0
|
|
|
|
|
0
|
unshift( @elts, @{ $elt->{chorus} } ); |
|
0
|
|
|
|
|
0
|
|
306
|
|
|
|
|
|
|
} |
307
|
|
|
|
|
|
|
elsif ( $rechorus->{type} && $rechorus->{tag} ) { |
308
|
0
|
|
|
|
|
0
|
push( @s, "{".$rechorus->{type}.": ".$rechorus->{tag}."}" ); |
309
|
|
|
|
|
|
|
} |
310
|
|
|
|
|
|
|
else { |
311
|
0
|
|
|
|
|
0
|
push( @s, "{chorus}" ); |
312
|
|
|
|
|
|
|
} |
313
|
0
|
|
|
|
|
0
|
next; |
314
|
|
|
|
|
|
|
} |
315
|
|
|
|
|
|
|
|
316
|
96
|
50
|
|
|
|
214
|
if ( $elt->{type} eq "tab" ) { |
317
|
0
|
0
|
|
|
|
0
|
push(@s, "") if $tidy; |
318
|
0
|
|
|
|
|
0
|
push(@s, "{start_of_tab}"); |
319
|
0
|
|
|
|
|
0
|
push(@s, @{$elt->{body}}); |
|
0
|
|
|
|
|
0
|
|
320
|
0
|
|
|
|
|
0
|
push(@s, "{end_of_tab}"); |
321
|
0
|
0
|
|
|
|
0
|
push(@s, "") if $tidy; |
322
|
0
|
|
|
|
|
0
|
next; |
323
|
|
|
|
|
|
|
} |
324
|
|
|
|
|
|
|
|
325
|
96
|
100
|
|
|
|
307
|
if ( $elt->{type} =~ /^comment(?:_italic|_box)?$/ ) { |
326
|
25
|
|
|
|
|
54
|
my $type = $elt->{type}; |
327
|
25
|
100
|
|
|
|
80
|
my $text = $expand ? $elt->{text} : $elt->{orig}; |
328
|
25
|
50
|
|
|
|
71
|
if ( $msp ) { |
329
|
0
|
0
|
|
|
|
0
|
$type = $type eq 'comment' |
|
|
0
|
|
|
|
|
|
330
|
|
|
|
|
|
|
? 'highlight' |
331
|
|
|
|
|
|
|
: $type eq 'comment_italic' |
332
|
|
|
|
|
|
|
? 'comment' |
333
|
|
|
|
|
|
|
: $type; |
334
|
|
|
|
|
|
|
} |
335
|
|
|
|
|
|
|
# Flatten chords/phrases. |
336
|
25
|
50
|
|
|
|
64
|
if ( $elt->{chords} ) { |
337
|
0
|
|
|
|
|
0
|
$text = ""; |
338
|
0
|
|
|
|
|
0
|
for ( 0..$#{ $elt->{chords} } ) { |
|
0
|
|
|
|
|
0
|
|
339
|
|
|
|
|
|
|
$text .= "[" . $elt->{chords}->[$_] . "]" |
340
|
0
|
0
|
|
|
|
0
|
if $elt->{chords}->[$_] ne ""; |
341
|
0
|
|
|
|
|
0
|
$text .= $elt->{phrases}->[$_]; |
342
|
|
|
|
|
|
|
} |
343
|
|
|
|
|
|
|
} |
344
|
25
|
50
|
|
|
|
66
|
$text = fmt_subst( $s, $text ) if $msp; |
345
|
25
|
50
|
|
|
|
60
|
push(@s, "") if $tidy; |
346
|
25
|
|
|
|
|
120
|
push(@s, "{$type: ".fq($text)."}"); |
347
|
25
|
50
|
|
|
|
80
|
push(@s, "") if $tidy; |
348
|
25
|
|
|
|
|
59
|
next; |
349
|
|
|
|
|
|
|
} |
350
|
|
|
|
|
|
|
|
351
|
71
|
50
|
|
|
|
161
|
if ( $elt->{type} eq "image" ) { |
352
|
0
|
|
|
|
|
0
|
my $uri = $elt->{uri}; |
353
|
0
|
0
|
0
|
|
|
0
|
if ( $msp && $uri !~ /^id=/ ) { |
354
|
0
|
|
0
|
|
|
0
|
$imgs{$uri} //= keys(%imgs); |
355
|
0
|
|
|
|
|
0
|
$uri = sprintf("id=img%02d", $imgs{$uri}); |
356
|
|
|
|
|
|
|
} |
357
|
0
|
|
|
|
|
0
|
my @args = ( "image:", $uri ); |
358
|
0
|
|
|
|
|
0
|
while ( my($k,$v) = each( %{ $elt->{opts} } ) ) { |
|
0
|
|
|
|
|
0
|
|
359
|
0
|
|
|
|
|
0
|
push( @args, "$k=$v" ); |
360
|
|
|
|
|
|
|
} |
361
|
0
|
|
|
|
|
0
|
foreach ( @args ) { |
362
|
0
|
0
|
|
|
|
0
|
next unless /\s/; |
363
|
0
|
|
|
|
|
0
|
$_ = '"' . $_ . '"'; |
364
|
|
|
|
|
|
|
} |
365
|
0
|
|
|
|
|
0
|
push( @s, "{@args}" ); |
366
|
0
|
|
|
|
|
0
|
next; |
367
|
|
|
|
|
|
|
} |
368
|
|
|
|
|
|
|
|
369
|
71
|
100
|
|
|
|
209
|
if ( $elt->{type} eq "diagrams" ) { |
370
|
2
|
50
|
|
|
|
18
|
$dumphdr = 0 unless $elt->{origin} eq "__CLI__"; |
371
|
|
|
|
|
|
|
push( @s, |
372
|
2
|
|
|
|
|
4
|
@{ ChordPro::Chords::list_chords |
373
|
|
|
|
|
|
|
( [ map { |
374
|
|
|
|
|
|
|
$s->{chordsinfo}->{$_}->{origin} eq 'inline' |
375
|
|
|
|
|
|
|
? $s->{chordsinfo}->{$_} |
376
|
|
|
|
|
|
|
: $s->{chordsinfo}->{$_}->{name} |
377
|
2
|
50
|
|
|
|
24
|
} @{$elt->{chords}} ], |
|
2
|
|
|
|
|
7
|
|
378
|
|
|
|
|
|
|
$elt->{origin}, |
379
|
2
|
|
|
|
|
4
|
$dumphdr ) } ); |
380
|
2
|
|
|
|
|
5
|
$dumphdr = 0; |
381
|
2
|
|
|
|
|
5
|
next; |
382
|
|
|
|
|
|
|
} |
383
|
|
|
|
|
|
|
|
384
|
69
|
100
|
|
|
|
163
|
if ( $elt->{type} eq "set" ) { |
385
|
30
|
50
|
|
|
|
151
|
if ( $elt->{name} eq "lyrics-only" ) { |
|
|
50
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
386
|
|
|
|
|
|
|
$lyrics_only = $elt->{value} |
387
|
0
|
0
|
|
|
|
0
|
unless $lyrics_only > 1; |
388
|
|
|
|
|
|
|
} |
389
|
|
|
|
|
|
|
elsif ( $elt->{name} eq "transpose" ) { |
390
|
|
|
|
|
|
|
} |
391
|
|
|
|
|
|
|
# Arbitrary config values. |
392
|
|
|
|
|
|
|
elsif ( $elt->{name} =~ /^(chordpro\..+)/ ) { |
393
|
0
|
|
|
|
|
0
|
my @k = split( /[.]/, $1 ); |
394
|
0
|
|
|
|
|
0
|
my $cc = {}; |
395
|
0
|
|
|
|
|
0
|
my $c = \$cc; |
396
|
0
|
|
|
|
|
0
|
foreach ( @k ) { |
397
|
0
|
|
|
|
|
0
|
$c = \($$c->{$_}); |
398
|
|
|
|
|
|
|
} |
399
|
0
|
|
|
|
|
0
|
$$c = $elt->{value}; |
400
|
0
|
|
|
|
|
0
|
$config->augment($cc); |
401
|
0
|
|
|
|
|
0
|
upd_config(); |
402
|
|
|
|
|
|
|
} |
403
|
30
|
|
|
|
|
77
|
next; |
404
|
|
|
|
|
|
|
} |
405
|
|
|
|
|
|
|
|
406
|
39
|
50
|
|
|
|
87
|
if ( $elt->{type} eq "ignore" ) { |
407
|
39
|
|
|
|
|
83
|
push( @s, $elt->{text} ); |
408
|
39
|
|
|
|
|
75
|
next; |
409
|
|
|
|
|
|
|
} |
410
|
|
|
|
|
|
|
|
411
|
|
|
|
|
|
|
} |
412
|
|
|
|
|
|
|
|
413
|
26
|
100
|
|
|
|
112
|
push(@s, "{end_of_$ctx}") if $ctx; |
414
|
|
|
|
|
|
|
|
415
|
|
|
|
|
|
|
# Process image assets. |
416
|
26
|
|
|
|
|
143
|
foreach ( sort { $imgs{$a} <=> $imgs{$b} } keys %imgs ) { |
|
0
|
|
|
|
|
0
|
|
417
|
0
|
|
|
|
|
0
|
my $url = $_; |
418
|
0
|
|
|
|
|
0
|
my $id = $imgs{$url}; |
419
|
0
|
|
|
|
|
0
|
my $type = "jpg"; |
420
|
0
|
0
|
|
|
|
0
|
$type = lc($1) if $url =~ /\.(\w+)$/; |
421
|
0
|
|
|
|
|
0
|
require MIME::Base64; |
422
|
0
|
|
|
|
|
0
|
require Image::Info; |
423
|
|
|
|
|
|
|
|
424
|
|
|
|
|
|
|
# Slurp the image. |
425
|
0
|
|
|
|
|
0
|
my $fd; |
426
|
0
|
0
|
|
|
|
0
|
unless ( open( $fd, '<:raw', $url ) ) { |
427
|
0
|
|
|
|
|
0
|
warn("$url: $!\n"); |
428
|
0
|
|
|
|
|
0
|
next; |
429
|
|
|
|
|
|
|
} |
430
|
0
|
|
|
|
|
0
|
my $data = do { local $/; <$fd> }; |
|
0
|
|
|
|
|
0
|
|
|
0
|
|
|
|
|
0
|
|
431
|
0
|
|
|
|
|
0
|
close($fd); |
432
|
|
|
|
|
|
|
|
433
|
|
|
|
|
|
|
# Get info. |
434
|
0
|
|
|
|
|
0
|
my $info = Image::Info::image_info(\$data); |
435
|
0
|
0
|
|
|
|
0
|
if ( $info->{error} ) { |
436
|
0
|
|
|
|
|
0
|
do_warn($info->{error}); |
437
|
0
|
|
|
|
|
0
|
next; |
438
|
|
|
|
|
|
|
} |
439
|
|
|
|
|
|
|
|
440
|
|
|
|
|
|
|
# Write in-line data. |
441
|
|
|
|
|
|
|
push( @s, |
442
|
|
|
|
|
|
|
sprintf( "##image: id=img%02d" . |
443
|
|
|
|
|
|
|
" src=%s type=%s width=%d height=%d enc=base64", |
444
|
|
|
|
|
|
|
$id, $url, $info->{file_ext}, |
445
|
0
|
|
|
|
|
0
|
$info->{width}, $info->{height} ) ); |
446
|
0
|
|
|
|
|
0
|
$data = MIME::Base64::encode($data, ''); |
447
|
0
|
|
|
|
|
0
|
my $i = 0; |
448
|
|
|
|
|
|
|
# Note: 76 is the standard chunk size for base64 data. |
449
|
0
|
|
|
|
|
0
|
while ( $i < length($data) ) { |
450
|
0
|
|
|
|
|
0
|
push( @s, "# ".substr($data, $i, 76) ); |
451
|
0
|
|
|
|
|
0
|
$i += 76; |
452
|
|
|
|
|
|
|
} |
453
|
|
|
|
|
|
|
} |
454
|
|
|
|
|
|
|
|
455
|
26
|
|
|
|
|
273
|
\@s; |
456
|
|
|
|
|
|
|
} |
457
|
|
|
|
|
|
|
|
458
|
|
|
|
|
|
|
sub songline { |
459
|
119
|
|
|
119
|
0
|
243
|
my ( $song, $elt ) = @_; |
460
|
|
|
|
|
|
|
|
461
|
119
|
100
|
100
|
|
|
467
|
if ( $lyrics_only || !exists($elt->{chords}) ) { |
462
|
40
|
|
|
|
|
61
|
return fq(join( "", @{ $elt->{phrases} } )); |
|
40
|
|
|
|
|
165
|
|
463
|
|
|
|
|
|
|
} |
464
|
|
|
|
|
|
|
|
465
|
79
|
|
|
|
|
153
|
my $line = ""; |
466
|
79
|
|
|
|
|
147
|
foreach my $c ( 0..$#{$elt->{chords}} ) { |
|
79
|
|
|
|
|
315
|
|
467
|
226
|
|
|
|
|
607
|
$line .= "[" . fq(chord( $song, $elt->{chords}->[$c])) . "]" . fq($elt->{phrases}->[$c]); |
468
|
|
|
|
|
|
|
} |
469
|
79
|
|
|
|
|
385
|
$line =~ s/^\[\]//; |
470
|
79
|
|
|
|
|
257
|
$line; |
471
|
|
|
|
|
|
|
} |
472
|
|
|
|
|
|
|
|
473
|
|
|
|
|
|
|
sub gridline { |
474
|
4
|
|
|
4
|
0
|
11
|
my ( $song, $elt ) = @_; |
475
|
|
|
|
|
|
|
|
476
|
4
|
|
|
|
|
8
|
my $line = ""; |
477
|
4
|
|
|
|
|
7
|
for ( @{ $elt->{tokens} } ) { |
|
4
|
|
|
|
|
12
|
|
478
|
44
|
100
|
|
|
|
111
|
$line .= " " if $line; |
479
|
44
|
100
|
|
|
|
104
|
if ( $_->{class} eq "chord" ) { |
480
|
10
|
|
|
|
|
25
|
$line .= chord( $song, $_->{chord} ); |
481
|
|
|
|
|
|
|
} |
482
|
|
|
|
|
|
|
else { |
483
|
34
|
|
|
|
|
67
|
$line .= $_->{symbol}; |
484
|
|
|
|
|
|
|
} |
485
|
|
|
|
|
|
|
} |
486
|
|
|
|
|
|
|
|
487
|
4
|
50
|
|
|
|
14
|
if ( $elt->{comment} ) { |
488
|
0
|
0
|
|
|
|
0
|
$line .= " " if $line; |
489
|
0
|
|
|
|
|
0
|
my $res = ""; |
490
|
0
|
|
|
|
|
0
|
my $t = $elt->{comment}; |
491
|
0
|
0
|
|
|
|
0
|
if ( $t->{chords} ) { |
492
|
0
|
|
|
|
|
0
|
for ( 0..$#{ $t->{chords} } ) { |
|
0
|
|
|
|
|
0
|
|
493
|
0
|
|
|
|
|
0
|
$res .= "[" . fq(chord( $song, $t->{chords}->[$_])) . "]" . fq($t->{phrases}->[$_]); |
494
|
|
|
|
|
|
|
} |
495
|
|
|
|
|
|
|
} |
496
|
|
|
|
|
|
|
else { |
497
|
0
|
|
|
|
|
0
|
$res .= fq($t->{text}); |
498
|
|
|
|
|
|
|
} |
499
|
0
|
|
|
|
|
0
|
$res =~ s/^\[\]//; |
500
|
0
|
|
|
|
|
0
|
$line .= $res; |
501
|
|
|
|
|
|
|
} |
502
|
|
|
|
|
|
|
|
503
|
4
|
|
|
|
|
10
|
$line; |
504
|
|
|
|
|
|
|
} |
505
|
|
|
|
|
|
|
|
506
|
|
|
|
|
|
|
sub chord { |
507
|
236
|
|
|
236
|
0
|
488
|
my ( $s, $c ) = @_; |
508
|
236
|
100
|
|
|
|
735
|
return "" unless length($c); |
509
|
176
|
|
|
|
|
508
|
local $c->info->{display} = undef; |
510
|
176
|
|
|
|
|
472
|
local $c->info->{format} = undef; |
511
|
176
|
|
|
|
|
502
|
my $t = $c->chord_display; |
512
|
176
|
50
|
|
|
|
2536
|
if ( $variant ne 'msp' ) { |
513
|
176
|
|
|
|
|
525
|
$t = demarkup($t); |
514
|
|
|
|
|
|
|
} |
515
|
176
|
100
|
|
|
|
626
|
return "*$t" if $c->info->is_annotation; |
516
|
174
|
|
|
|
|
717
|
return $t; |
517
|
|
|
|
|
|
|
} |
518
|
|
|
|
|
|
|
|
519
|
|
|
|
|
|
|
1; |