8 my ($filename, @others) = @ARGV;
9 my $opus = MIDI::Opus->new({from_file => $filename});
17 channel_after_touch => 2,
18 pitch_wheel_change => 2
22 set_tempo => sub { 1; },
23 note_off => sub { 1; },
24 note_on => sub { 1; },
25 control_change => sub { $_[3] == 64; },
31 return map { [$_->[0], $t += $_->[1], @{$_}[2..(@$_-1)]]; } @_;
37 return map { my $tsave = $t; $t = $_->[1]; [$_->[0], $t - $tsave, @{$_}[2..(@$_-1)]]; } @_;
42 return reltime grep { ($isclean{$_->[0]} // sub { 0; })->(@$_) } abstime @_;
47 my $opus2 = MIDI::Opus->new({from_file => $_});
48 if($opus2->ticks() != $opus->ticks())
50 my $tickfactor = $opus->ticks() / $opus2->ticks();
53 $_->events(reltime map { $_->[1] = int($_->[1] * $tickfactor + 0.5); $_; } abstime $_->events());
56 $opus->tracks($opus->tracks(), $opus2->tracks());
62 my @arg = split /\s+/, $_;
64 print "Executing: $cmd @arg\n";
67 my $tracks = $opus->tracks_r();
68 $tracks->[$_]->events_r([clean($tracks->[$_]->events())])
73 print $opus->dump({ dump_tracks => 1 });
75 elsif($cmd eq 'ticks')
79 $opus->ticks($arg[0]);
83 print "Ticks: ", $opus->ticks(), "\n";
86 elsif($cmd eq 'tricks')
88 print "haha, very funny\n";
90 elsif($cmd eq 'retrack')
92 my $tracks = $opus->tracks_r();
96 for(abstime $tracks->[$_]->events())
98 my $p = $chanpos{$_->[0]};
101 my $c = $_->[$p] + 1;
102 push @{$newtracks[$c]}, $_;
106 push @{$newtracks[0]}, $_;
110 $opus->tracks_r([map { ($_ && @$_) ? MIDI::Track->new({ events => [reltime @$_] }) : () } @newtracks]);
112 elsif($cmd eq 'program')
114 my $tracks = $opus->tracks_r();
115 my ($channel, $program) = @arg;
120 for(abstime $tracks->[$_]->events())
122 my $p = $chanpos{$_->[0]};
125 my $c = $_->[$p] + 1;
129 if $_->[0] eq 'patch_change';
132 push @events, ['patch_change', $_->[1], $c-1, $program-1]
140 $tracks->[$_]->events_r([reltime @events]);
143 elsif($cmd eq 'transpose')
145 my $tracks = $opus->tracks_r();
146 my ($track, $channel, $delta) = @arg;
147 for(($track eq '*') ? (0..@$tracks-1) : $track)
149 for($tracks->[$_]->events())
151 my $p = $chanpos{$_->[0]};
154 my $c = $_->[$p] + 1;
155 if($channel eq '*' || $c == $channel)
157 if($_->[0] eq 'note_on' || $_->[0] eq 'note_off')
166 elsif($cmd eq 'percussion')
168 my $tracks = $opus->tracks_r();
173 for(abstime $tracks->[$_]->events())
175 my $p = $chanpos{$_->[0]};
178 my $c = $_->[$p] + 1;
181 if($_->[0] eq 'note_on' || $_->[0] eq 'note_off')
183 if(length $map{$_->[3]})
185 $_->[3] = $map{$_->[3]};
187 elsif(exists $map{$_->[3]})
196 $tracks->[$_]->events_r([reltime @events]);
199 elsif($cmd eq 'tracks')
201 my $tracks = $opus->tracks_r();
214 next if $taken{$_}++ and not $force;
215 push @t, $tracks->[$_];
217 $opus->tracks_r(\@t);
230 for($tracks->[$_]->events())
233 $_->[0] = 'note_off' if $_->[0] eq 'note_on' and $_->[4] == 0;
235 my $p = $chanpos{$_->[0]};
238 my $c = $_->[$p] + 1;
239 $channels{$c} //= {};
240 if($_->[0] eq 'patch_change')
242 ++$channels{$c}{$_->[3]};
245 ++$notes if $_->[0] eq 'note_on';
246 $notehash{$_->[2]}{$_->[3]} = $t if $_->[0] eq 'note_on';
247 $notehash{$_->[2]}{$_->[3]} = undef if $_->[0] eq 'note_off';
248 $name = $_->[2] if $_->[0] eq 'track_name';
250 my $channels = join " ", map { sprintf "%s(%s)", $_, join ",", sort { $a <=> $b } keys %{$channels{$_}} } sort { $a <=> $b } keys %channels;
252 while(my ($k1, $v1) = each %notehash)
254 while(my ($k2, $v2) = each %$v1)
256 push @stuck, sprintf "%d:%d@%.1f%%", $k1+1, $k2, $v2 * 100.0 / $t
260 print " $name" if defined $name;
261 print " (channel $channels)" if $channels ne "";
262 print " ($events events)" if $events;
263 print " ($notes notes)" if $notes;
264 print " (notes @stuck stuck)" if @stuck;
269 elsif($cmd eq 'save')
271 $opus->write_to_file($arg[0]);
275 print "Unknown command, allowed commands:\n";
278 print " ticks [value]\n";
280 print " program <channel> <program (1-based)>\n";
281 print " transpose <track|*> <channel|*> <delta>\n";
282 print " percussion <from> <to> [<from> <to> ...]\n";
283 print " tracks [trackno] [trackno] ...\n";
285 print "Done with: $cmd @arg\n";