blob: 85f56b3beb9f4be6a118eba552e1b5e6d3198d7f [file] [log] [blame]
Guido van Rossumf94e3091992-04-22 13:24:13 +00001# Class interface to the CD module.
2
3import cd, CD
4
5Error = 'Readcd.Error'
6_Stop = 'Readcd.Stop'
7
8def _doatime(self, type, data):
9 if ((data[0] * 60) + data[1]) * 75 + data[2] > self.end:
Guido van Rossumada67ec1992-06-03 16:47:24 +000010## print 'done with list entry',`self.listindex`
Guido van Rossumf94e3091992-04-22 13:24:13 +000011 raise _Stop
12 func, arg = self.callbacks[type]
13 if func:
14 func(arg, type, data)
15
16def _dopnum(self, type, data):
17 if data > self.end:
Guido van Rossumada67ec1992-06-03 16:47:24 +000018## print 'done with list entry',`self.listindex`
Guido van Rossumf94e3091992-04-22 13:24:13 +000019 raise _Stop
20 func, arg = self.callbacks[type]
21 if func:
22 func(arg, type, data)
23
24class Readcd():
25 def init(self, *arg):
26 if len(arg) == 0:
27 self.player = cd.open()
28 elif len(arg) == 1:
29 self.player = cd.open(arg[0])
30 elif len(arg) == 2:
31 self.player = cd.open(arg[0], arg[1])
32 else:
33 raise Error, 'bad init call'
34 self.list = []
35 self.callbacks = [(None, None)] * 8
36 self.parser = cd.createparser()
37 self.playing = 0
38 self.end = 0
39 self.status = None
40 self.trackinfo = None
41 return self
42
43 def eject(self):
44 self.player.eject()
45 self.list = []
46 self.end = 0
47 self.listindex = 0
48 self.status = None
49 self.trackinfo = None
50 if self.playing:
Guido van Rossumada67ec1992-06-03 16:47:24 +000051## print 'stop playing from eject'
Guido van Rossumf94e3091992-04-22 13:24:13 +000052 raise _Stop
53
54 def pmsf2msf(self, track, min, sec, frame):
55 if not self.status:
Guido van Rossum58230c31992-04-22 14:33:26 +000056 self.cachestatus()
Guido van Rossumf94e3091992-04-22 13:24:13 +000057 if track < self.status[5] or track > self.status[6]:
58 raise Error, 'track number out of range'
Guido van Rossum58230c31992-04-22 14:33:26 +000059 if not self.trackinfo:
60 self.cacheinfo()
Guido van Rossumf94e3091992-04-22 13:24:13 +000061 start, total = self.trackinfo[track]
62 start = ((start[0] * 60) + start[1]) * 75 + start[2]
63 total = ((total[0] * 60) + total[1]) * 75 + total[2]
64 block = ((min * 60) + sec) * 75 + frame
65 if block > total:
66 raise Error, 'out of range'
67 block = start + block
68 min, block = divmod(block, 75*60)
69 sec, frame = divmod(block, 75)
70 return min, sec, frame
71
72 def reset(self):
73 self.list = []
74
75 def appendtrack(self, track):
76 self.appendstretch(track, track)
77
78 def appendstretch(self, start, end):
79 if not self.status:
Guido van Rossum58230c31992-04-22 14:33:26 +000080 self.cachestatus()
Guido van Rossumf94e3091992-04-22 13:24:13 +000081 if not start:
82 start = 1
83 if not end:
84 end = self.status[6]
85 try:
86 l = len(end)
87 if l == 4:
88 prog, min, sec, frame = end
89 if prog < self.status[5] or prog > self.status[6]:
90 raise Error, 'range error'
91 end = self.pmsf2msf(prog, min, sec, frame)
92 elif l <> 3:
93 raise Error, 'syntax error'
94 except TypeError:
95 if end < self.status[5] or end > self.status[6]:
96 raise Error, 'range error'
97 try:
98 l = len(start)
99 if l == 4:
100 prog, min, sec, frame = start
101 if prog < self.status[5] or prog > self.status[6]:
102 raise Error, 'range error'
103 start = self.pmsf2msf(prog, min, sec, frame)
104 elif l <> 3:
105 raise Error, 'syntax error'
106 except TypeError:
107 if start < self.status[5] or start > self.status[6]:
108 raise Error, 'range error'
109 if len(self.list) > 0:
110 s, e = self.list[-1]
111 try:
112 l = len(e)
113 except TypeError:
114 if start == e+1:
115 start = s
116 del self.list[-1]
117 self.list.append((start, end))
118
119 def settracks(self, list):
120 self.list = []
121 for track in list:
122 self.appendtrack(track)
123
124 def setcallback(self, type, func, arg):
125 if type < 0 or type >= 8:
126 raise Error, 'type out of range'
127 self.callbacks[type] = (func, arg)
128 if self.playing:
129 try:
130 l = len(end)
131 if type <> CD.ATIME:
132 self.parser.setcallback(type, func, arg)
133 except TypeError:
134 if type <> CD.PNUM:
135 self.parser.setcallback(type, func, arg)
136
137 def removecallback(self, type):
138 if type < 0 or type >= 8:
139 raise Error, 'type out of range'
140 self.callbacks[type] = (None, None)
141 if self.playing:
142 try:
143 l = len(end)
144 if type <> CD.ATIME:
145 self.parser.removecallback(type)
146 except TypeError:
147 if type <> CD.PNUM:
148 self.parser.removecallback(type)
149
150 def gettrackinfo(self, *arg):
151 if not self.status:
Guido van Rossum58230c31992-04-22 14:33:26 +0000152 self.cachestatus()
Guido van Rossumf94e3091992-04-22 13:24:13 +0000153 if not self.trackinfo:
Guido van Rossum58230c31992-04-22 14:33:26 +0000154 self.cacheinfo()
Guido van Rossumf94e3091992-04-22 13:24:13 +0000155 if len(arg) == 0:
156 return self.trackinfo[self.status[5]:self.status[6]+1]
157 result = []
158 for i in arg:
159 if i < self.status[5] or i > self.status[6]:
160 raise Error, 'range error'
161 result.append(self.trackinfo[i])
162 return result
163
Guido van Rossum58230c31992-04-22 14:33:26 +0000164 def cacheinfo(self):
165 if not self.status:
166 self.cachestatus()
167 self.trackinfo = []
168 for i in range(self.status[5]):
169 self.trackinfo.append(None)
170 for i in range(self.status[5], self.status[6]+1):
171 self.trackinfo.append(self.player.gettrackinfo(i))
172
173 def cachestatus(self):
Guido van Rossumdee5d8e1992-04-22 13:44:48 +0000174 self.status = self.player.getstatus()
Guido van Rossuma3cd0711992-04-23 11:34:39 +0000175 if self.status[0] == CD.NODISC:
Guido van Rossum58230c31992-04-22 14:33:26 +0000176 self.status = None
Guido van Rossuma3cd0711992-04-23 11:34:39 +0000177 raise Error, 'no disc in player'
Guido van Rossum58230c31992-04-22 14:33:26 +0000178
179 def getstatus(self):
180 return self.player.getstatus()
Guido van Rossumf94e3091992-04-22 13:24:13 +0000181
182 def play(self):
183 if not self.status:
Guido van Rossum58230c31992-04-22 14:33:26 +0000184 self.cachestatus()
Guido van Rossumf94e3091992-04-22 13:24:13 +0000185 size = self.player.bestreadsize()
186 self.listindex = 0
187 self.playing = 0
188 for i in range(8):
189 func, arg = self.callbacks[i]
190 if func:
191 self.parser.setcallback(i, func, arg)
192 else:
193 self.parser.removecallback(i)
194 if len(self.list) == 0:
195 for i in range(self.status[5], self.status[6]+1):
196 self.appendtrack(i)
197 while 1:
198 if not self.playing:
199 if self.listindex >= len(self.list):
200 return
201 start, end = self.list[self.listindex]
Guido van Rossumada67ec1992-06-03 16:47:24 +0000202## print 'starting with',`(start, end)`
Guido van Rossumf94e3091992-04-22 13:24:13 +0000203 try:
204 min, sec, frame = start
205 dummy = self.player.seek(min, sec, frame)
206 except TypeError:
207 dummy = self.player.seektrack(start)
208 try:
209 min, sec, frame = end
210 self.parser.setcallback(CD.ATIME, _doatime, self)
211 self.end = (min * 60 + sec) * 75 + frame
212 func, arg = self.callbacks[CD.PNUM]
213 if func:
214 self.parser.setcallback(CD.PNUM, func, arg)
215 else:
216 self.parser.removecallback(CD.PNUM)
217 except TypeError:
218 self.parser.setcallback(CD.PNUM, _dopnum, self)
219 self.end = end
220 func, arg = self.callbacks[CD.ATIME]
221 if func:
222 self.parser.setcallback(CD.ATIME, func, arg)
223 else:
224 self.parser.removecallback(CD.ATIME)
225 self.playing = 1
226 data = self.player.readda(size)
227 if data == '':
228 self.playing = 0
229 self.listindex = self.listindex + 1
230 continue
231 try:
232 self.parser.parseframe(data)
233 except _Stop:
234 self.playing = 0
235 self.listindex = self.listindex + 1