1 | #!/usr/bin/perl -w |
---|
2 | use strict; |
---|
3 | |
---|
4 | use IPC::Open2; |
---|
5 | use IO::Prompt; |
---|
6 | |
---|
7 | my $FLAC_FILE = shift; |
---|
8 | my $TRACK = shift || 1; |
---|
9 | |
---|
10 | |
---|
11 | # get the track start positions from the embedded cuesheet |
---|
12 | open my $CUESHEET, "metaflac --export-cuesheet-to - $FLAC_FILE |"; |
---|
13 | |
---|
14 | my @position_for_track; |
---|
15 | my $i = 1; |
---|
16 | while (<$CUESHEET>) { |
---|
17 | if (my ($m,$s,$f) = /INDEX 01 (\d\d):(\d\d):(\d\d)/) { |
---|
18 | $position_for_track[$i] = ($m * 60) + $s + ($f / 75); |
---|
19 | $i++; |
---|
20 | } |
---|
21 | } |
---|
22 | |
---|
23 | close $CUESHEET; |
---|
24 | |
---|
25 | |
---|
26 | my $track_count = scalar @position_for_track - 1; |
---|
27 | print "$track_count tracks\n"; |
---|
28 | |
---|
29 | die "There is no track $TRACK" if $TRACK < 1 or $TRACK > $track_count; |
---|
30 | |
---|
31 | |
---|
32 | # launch the flac123 player as a daemon |
---|
33 | my $flac123_pid = open2(my $PLAYER_OUT, my $PLAYER_IN, qw{flac123 -R}); |
---|
34 | |
---|
35 | print "flac123 PID = $flac123_pid\n"; |
---|
36 | |
---|
37 | my $current_track = $TRACK; |
---|
38 | my $next_track_start = $position_for_track[$current_track + 1]; |
---|
39 | |
---|
40 | print "Track $current_track\n"; |
---|
41 | |
---|
42 | my %commands = ( |
---|
43 | z => sub { |
---|
44 | return if $current_track <= 1; |
---|
45 | $current_track--; |
---|
46 | my $position = $position_for_track[$current_track]; |
---|
47 | print $PLAYER_IN "JUMP $position\n"; |
---|
48 | print "Track $current_track\n"; |
---|
49 | }, |
---|
50 | x => sub {}, |
---|
51 | c => sub {}, |
---|
52 | v => sub {}, |
---|
53 | b => sub { |
---|
54 | return if $current_track >= $track_count; |
---|
55 | # note, don't increment or print $current_track here, |
---|
56 | # since it will be automatically detected by the other |
---|
57 | # process watching the output of flac123 |
---|
58 | my $position = $position_for_track[$current_track + 1]; |
---|
59 | print $PLAYER_IN "JUMP $position\n"; |
---|
60 | }, |
---|
61 | q => sub { |
---|
62 | print $PLAYER_IN "QUIT\n"; |
---|
63 | exit; |
---|
64 | }, |
---|
65 | ); |
---|
66 | |
---|
67 | # track has advanced automatically |
---|
68 | $SIG{USR1} = sub { |
---|
69 | # detect track change |
---|
70 | my $time = read_time(); |
---|
71 | if (defined $next_track_start && $time >= $next_track_start) { |
---|
72 | $current_track++; |
---|
73 | $next_track_start = $position_for_track[$current_track + 1]; |
---|
74 | print "Track $current_track\n"; |
---|
75 | } |
---|
76 | }; |
---|
77 | |
---|
78 | my $parent_pid = $$; |
---|
79 | my $pid = fork; |
---|
80 | |
---|
81 | |
---|
82 | if ($pid) { |
---|
83 | # parent |
---|
84 | while (my $cmd = prompt('', -one_char, -echo => '')) { |
---|
85 | if (exists $commands{$cmd}) { |
---|
86 | $commands{$cmd}->(); |
---|
87 | } |
---|
88 | } |
---|
89 | |
---|
90 | } elsif (defined $pid) { |
---|
91 | # child |
---|
92 | |
---|
93 | print $PLAYER_IN "LOAD $FLAC_FILE\n"; |
---|
94 | print $PLAYER_IN "JUMP $position_for_track[$TRACK]\n"; |
---|
95 | |
---|
96 | while (<$PLAYER_OUT>) { |
---|
97 | if (/\@F (\d+) (\d+) (\d+\.\d\d) (\d+\.\d\d)/) { |
---|
98 | my ($frame, $frames_left, $time, $time_left) = ($1, $2, $3, $4); |
---|
99 | # note the current time and alert the parent |
---|
100 | write_time($time); |
---|
101 | kill 'USR1', $parent_pid; |
---|
102 | } |
---|
103 | } |
---|
104 | waitpid $flac123_pid, 0; |
---|
105 | |
---|
106 | } else { |
---|
107 | die "Unable to fork: $!"; |
---|
108 | } |
---|
109 | |
---|
110 | sub write_time { |
---|
111 | my ($time) = @_; |
---|
112 | open my $TIME, '>', 'current_time'; |
---|
113 | print $TIME "$time\n"; |
---|
114 | close $TIME; |
---|
115 | } |
---|
116 | |
---|
117 | sub read_time { |
---|
118 | open my $TIME, '<', 'current_time'; |
---|
119 | my $time = <$TIME>; |
---|
120 | close $TIME; |
---|
121 | chomp $time; |
---|
122 | return $time; |
---|
123 | } |
---|
124 | |
---|
125 | =begin flac123 -R outputs |
---|
126 | |
---|
127 | @R FLAC123 |
---|
128 | flac123 tagline. Output at startup. |
---|
129 | |
---|
130 | @I ID3:<a><b><c><d><e><f> |
---|
131 | Prints out the metadata information after loading the flac file. |
---|
132 | a = title (30 chars) |
---|
133 | b = artist (30 chars) |
---|
134 | c = album (30 chars) |
---|
135 | d = year (4 chars) |
---|
136 | e = comment (30 chars) |
---|
137 | f = genre (30 chars) |
---|
138 | |
---|
139 | @I filename |
---|
140 | Prints out the filename of the flac file, minus the extension. Happens after |
---|
141 | a flac file has been loaded and there is no metadata available. |
---|
142 | |
---|
143 | @F <current-frame> <frames-remaining> <current-time> <time-remaining> |
---|
144 | Frame decoding status updates (once per frame). |
---|
145 | Current-frame and frames-remaining are integers; current-time and |
---|
146 | time-remaining floating point numbers with two decimal places. |
---|
147 | |
---|
148 | @P {0, 1, 2} |
---|
149 | Stop/pause status. |
---|
150 | 0 - playing has stopped. When 'STOP' is entered, or the flac file is finished. |
---|
151 | 1 - Playing is paused. Enter 'PAUSE' or 'P' to continue. |
---|
152 | 2 - Playing has begun again. |
---|
153 | |
---|
154 | =cut |
---|