Mercurial > audlegacy
annotate src/audacious/playback.c @ 3082:91df04ec5ef7
From: Daniel Drake <dsd@gentoo.org>
We don't need to check playback status 100 times per second. Just start
and stop the visualization when those operations actually happen.
author | William Pitcock <nenolod@atheme-project.org> |
---|---|
date | Mon, 16 Jul 2007 13:31:51 -0500 |
parents | 01ae56759d15 |
children | 20830a69b5f5 |
rev | line source |
---|---|
2313 | 1 /* Audacious - Cross-platform multimedia player |
2 * Copyright (C) 2005-2007 Audacious development team | |
3 * | |
4 * Based on BMP: | |
5 * Copyright (C) 2003-2004 BMP development team. | |
6 * | |
7 * Based on XMMS: | |
8 * Copyright (C) 1998-2003 XMMS development team. | |
9 * | |
10 * This program is free software; you can redistribute it and/or modify | |
11 * it under the terms of the GNU General Public License as published by | |
12 * the Free Software Foundation; under version 2 of the License. | |
13 * | |
14 * This program is distributed in the hope that it will be useful, | |
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
17 * GNU General Public License for more details. | |
18 * | |
19 * You should have received a copy of the GNU General Public License | |
20 * along with this program; if not, write to the Free Software | |
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | |
22 */ | |
23 | |
24 #ifdef HAVE_CONFIG_H | |
25 # include "config.h" | |
26 #endif | |
27 | |
28 #include <glib.h> | |
29 #include <glib/gi18n.h> | |
30 #include <glib/gprintf.h> | |
31 #include <stdlib.h> | |
32 #include <string.h> | |
33 #include <time.h> | |
34 | |
35 #include <unistd.h> | |
36 #include <sys/types.h> | |
37 #include <sys/stat.h> | |
38 #include <dirent.h> | |
39 | |
2717 | 40 #include "configdb.h" |
2313 | 41 |
2420 | 42 #include "hook.h" |
2313 | 43 #include "input.h" |
44 #include "main.h" | |
45 #include "ui_equalizer.h" | |
46 #include "output.h" | |
47 #include "playlist.h" | |
2420 | 48 #include "ui_main.h" |
2313 | 49 #include "ui_playlist.h" |
50 #include "ui_skinselector.h" | |
51 #include "util.h" | |
3082
91df04ec5ef7
From: Daniel Drake <dsd@gentoo.org>
William Pitcock <nenolod@atheme-project.org>
parents:
3042
diff
changeset
|
52 #include "visualization.h" |
2313 | 53 |
54 #include "playback.h" | |
55 | |
3018
6a9fdd5aee3a
Move timer updates out of the 100hz giant loop.
Daniel Drake <dsd@gentoo.org>
parents:
3001
diff
changeset
|
56 static int song_info_timeout_source = 0; |
2313 | 57 |
58 gint | |
59 playback_get_time(void) | |
60 { | |
2438
8750a62abed8
[svn] Provide flags in InputPlayback for common plugin flag needs, and provide a
iabervon
parents:
2436
diff
changeset
|
61 InputPlayback *playback; |
2420 | 62 g_return_val_if_fail(playback_get_playing(), -1); |
2438
8750a62abed8
[svn] Provide flags in InputPlayback for common plugin flag needs, and provide a
iabervon
parents:
2436
diff
changeset
|
63 playback = get_current_input_playback(); |
8750a62abed8
[svn] Provide flags in InputPlayback for common plugin flag needs, and provide a
iabervon
parents:
2436
diff
changeset
|
64 g_return_val_if_fail(playback, -1); |
2313 | 65 |
2438
8750a62abed8
[svn] Provide flags in InputPlayback for common plugin flag needs, and provide a
iabervon
parents:
2436
diff
changeset
|
66 if (playback->plugin->get_time) |
8750a62abed8
[svn] Provide flags in InputPlayback for common plugin flag needs, and provide a
iabervon
parents:
2436
diff
changeset
|
67 return playback->plugin->get_time(playback); |
8750a62abed8
[svn] Provide flags in InputPlayback for common plugin flag needs, and provide a
iabervon
parents:
2436
diff
changeset
|
68 if (playback->error) |
8750a62abed8
[svn] Provide flags in InputPlayback for common plugin flag needs, and provide a
iabervon
parents:
2436
diff
changeset
|
69 return -2; |
8750a62abed8
[svn] Provide flags in InputPlayback for common plugin flag needs, and provide a
iabervon
parents:
2436
diff
changeset
|
70 if (!playback->playing || |
8750a62abed8
[svn] Provide flags in InputPlayback for common plugin flag needs, and provide a
iabervon
parents:
2436
diff
changeset
|
71 (playback->eof && !playback->output->buffer_playing())) |
8750a62abed8
[svn] Provide flags in InputPlayback for common plugin flag needs, and provide a
iabervon
parents:
2436
diff
changeset
|
72 return -1; |
8750a62abed8
[svn] Provide flags in InputPlayback for common plugin flag needs, and provide a
iabervon
parents:
2436
diff
changeset
|
73 return playback->output->output_time(); |
2313 | 74 } |
75 | |
76 void | |
77 playback_initiate(void) | |
78 { | |
79 PlaylistEntry *entry = NULL; | |
80 Playlist *playlist = playlist_get_active(); | |
81 | |
2420 | 82 g_return_if_fail(playlist_get_length(playlist) != 0); |
2313 | 83 |
84 if (playback_get_playing()) | |
85 playback_stop(); | |
86 | |
3020 | 87 ui_vis_clear_data(mainwin_vis); |
2313 | 88 svis_clear_data(mainwin_svis); |
89 mainwin_disable_seekbar(); | |
90 | |
91 entry = playlist_get_entry_to_play(playlist); | |
2420 | 92 g_return_if_fail(entry != NULL); |
2592
6f7be3702c5f
[svn] - revert the penalty-based "find something to play" logic of 1.2. it's
nenolod
parents:
2454
diff
changeset
|
93 playback_play_file(entry); |
2313 | 94 |
95 if (playback_get_time() != -1) { | |
96 equalizerwin_load_auto_preset(entry->filename); | |
97 input_set_eq(cfg.equalizer_active, cfg.equalizer_preamp, | |
98 cfg.equalizer_bands); | |
99 output_set_eq(cfg.equalizer_active, cfg.equalizer_preamp, | |
100 cfg.equalizer_bands); | |
101 } | |
102 | |
103 playlist_check_pos_current(playlist); | |
104 mainwin_set_info_text(); | |
3018
6a9fdd5aee3a
Move timer updates out of the 100hz giant loop.
Daniel Drake <dsd@gentoo.org>
parents:
3001
diff
changeset
|
105 mainwin_update_song_info(); |
6a9fdd5aee3a
Move timer updates out of the 100hz giant loop.
Daniel Drake <dsd@gentoo.org>
parents:
3001
diff
changeset
|
106 |
6a9fdd5aee3a
Move timer updates out of the 100hz giant loop.
Daniel Drake <dsd@gentoo.org>
parents:
3001
diff
changeset
|
107 /* FIXME: use g_timeout_add_seconds when glib-2.14 is required */ |
6a9fdd5aee3a
Move timer updates out of the 100hz giant loop.
Daniel Drake <dsd@gentoo.org>
parents:
3001
diff
changeset
|
108 song_info_timeout_source = g_timeout_add(1000, |
6a9fdd5aee3a
Move timer updates out of the 100hz giant loop.
Daniel Drake <dsd@gentoo.org>
parents:
3001
diff
changeset
|
109 (GSourceFunc) mainwin_update_song_info, NULL); |
2407
1dc1d36d0347
[svn] - add hooks: playback begin, playback end, playlist reached end
nenolod
parents:
2366
diff
changeset
|
110 |
3000
bbca1e0e054a
don't call gtk_widget_show/hide from func that's called every now and then
Tomasz Mon <desowin@gmail.com>
parents:
2976
diff
changeset
|
111 if (cfg.player_shaded) { |
bbca1e0e054a
don't call gtk_widget_show/hide from func that's called every now and then
Tomasz Mon <desowin@gmail.com>
parents:
2976
diff
changeset
|
112 gtk_widget_show(mainwin_stime_min); |
bbca1e0e054a
don't call gtk_widget_show/hide from func that's called every now and then
Tomasz Mon <desowin@gmail.com>
parents:
2976
diff
changeset
|
113 gtk_widget_show(mainwin_stime_sec); |
3042
01ae56759d15
mainwin_sposition is now UiSkinnedHorizontalSlider
Tomasz Mon <desowin@gmail.com>
parents:
3041
diff
changeset
|
114 gtk_widget_show(mainwin_sposition); |
3001
6d4b7b739232
fully implement UiSkinnedNumber, number.c no longer needed
Tomasz Mon <desowin@gmail.com>
parents:
3000
diff
changeset
|
115 } else { |
6d4b7b739232
fully implement UiSkinnedNumber, number.c no longer needed
Tomasz Mon <desowin@gmail.com>
parents:
3000
diff
changeset
|
116 gtk_widget_show(mainwin_minus_num); |
6d4b7b739232
fully implement UiSkinnedNumber, number.c no longer needed
Tomasz Mon <desowin@gmail.com>
parents:
3000
diff
changeset
|
117 gtk_widget_show(mainwin_10min_num); |
6d4b7b739232
fully implement UiSkinnedNumber, number.c no longer needed
Tomasz Mon <desowin@gmail.com>
parents:
3000
diff
changeset
|
118 gtk_widget_show(mainwin_min_num); |
6d4b7b739232
fully implement UiSkinnedNumber, number.c no longer needed
Tomasz Mon <desowin@gmail.com>
parents:
3000
diff
changeset
|
119 gtk_widget_show(mainwin_10sec_num); |
6d4b7b739232
fully implement UiSkinnedNumber, number.c no longer needed
Tomasz Mon <desowin@gmail.com>
parents:
3000
diff
changeset
|
120 gtk_widget_show(mainwin_sec_num); |
3040
067e0186623a
import UiSkinnedHorizontalSlider
Tomasz Mon <desowin@gmail.com>
parents:
3021
diff
changeset
|
121 gtk_widget_show(mainwin_position); |
3000
bbca1e0e054a
don't call gtk_widget_show/hide from func that's called every now and then
Tomasz Mon <desowin@gmail.com>
parents:
2976
diff
changeset
|
122 } |
bbca1e0e054a
don't call gtk_widget_show/hide from func that's called every now and then
Tomasz Mon <desowin@gmail.com>
parents:
2976
diff
changeset
|
123 |
3082
91df04ec5ef7
From: Daniel Drake <dsd@gentoo.org>
William Pitcock <nenolod@atheme-project.org>
parents:
3042
diff
changeset
|
124 vis_playback_start(); |
91df04ec5ef7
From: Daniel Drake <dsd@gentoo.org>
William Pitcock <nenolod@atheme-project.org>
parents:
3042
diff
changeset
|
125 |
2407
1dc1d36d0347
[svn] - add hooks: playback begin, playback end, playlist reached end
nenolod
parents:
2366
diff
changeset
|
126 hook_call("playback begin", entry); |
2313 | 127 } |
128 | |
129 void | |
130 playback_pause(void) | |
131 { | |
132 if (!playback_get_playing()) | |
133 return; | |
134 | |
2436
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
135 if (!get_current_input_playback()) |
2313 | 136 return; |
137 | |
138 ip_data.paused = !ip_data.paused; | |
139 | |
2436
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
140 if (get_current_input_playback()->plugin->pause) |
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
141 get_current_input_playback()->plugin->pause(get_current_input_playback(), |
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
142 ip_data.paused); |
2313 | 143 |
3039
c981df78db8c
added hooks: playback pause and playback unpause
Giacomo Lozito <james@develia.org>
parents:
3021
diff
changeset
|
144 if (ip_data.paused) |
c981df78db8c
added hooks: playback pause and playback unpause
Giacomo Lozito <james@develia.org>
parents:
3021
diff
changeset
|
145 hook_call("playback pause", NULL); |
c981df78db8c
added hooks: playback pause and playback unpause
Giacomo Lozito <james@develia.org>
parents:
3021
diff
changeset
|
146 else |
c981df78db8c
added hooks: playback pause and playback unpause
Giacomo Lozito <james@develia.org>
parents:
3021
diff
changeset
|
147 hook_call("playback unpause", NULL); |
c981df78db8c
added hooks: playback pause and playback unpause
Giacomo Lozito <james@develia.org>
parents:
3021
diff
changeset
|
148 |
2313 | 149 g_return_if_fail(mainwin_playstatus != NULL); |
150 | |
151 if (ip_data.paused) | |
152 playstatus_set_status(mainwin_playstatus, STATUS_PAUSE); | |
153 else | |
154 playstatus_set_status(mainwin_playstatus, STATUS_PLAY); | |
155 } | |
156 | |
157 void | |
158 playback_stop(void) | |
159 { | |
2436
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
160 if (ip_data.playing && get_current_input_playback()) { |
2313 | 161 |
162 if (playback_get_paused()) { | |
163 output_flush(get_written_time()); /* to avoid noise */ | |
164 playback_pause(); | |
165 } | |
166 | |
167 ip_data.playing = FALSE; | |
168 | |
2436
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
169 if (get_current_input_playback()->plugin->stop) |
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
170 get_current_input_playback()->plugin->stop(get_current_input_playback()); |
2313 | 171 |
172 free_vis_data(); | |
173 ip_data.paused = FALSE; | |
174 | |
175 if (input_info_text) { | |
176 g_free(input_info_text); | |
177 input_info_text = NULL; | |
178 mainwin_set_info_text(); | |
179 } | |
2436
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
180 |
2438
8750a62abed8
[svn] Provide flags in InputPlayback for common plugin flag needs, and provide a
iabervon
parents:
2436
diff
changeset
|
181 g_free(get_current_input_playback()->filename); |
2436
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
182 g_free(get_current_input_playback()); |
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
183 set_current_input_playback(NULL); |
2313 | 184 } |
185 | |
186 ip_data.buffering = FALSE; | |
187 ip_data.playing = FALSE; | |
3018
6a9fdd5aee3a
Move timer updates out of the 100hz giant loop.
Daniel Drake <dsd@gentoo.org>
parents:
3001
diff
changeset
|
188 |
6a9fdd5aee3a
Move timer updates out of the 100hz giant loop.
Daniel Drake <dsd@gentoo.org>
parents:
3001
diff
changeset
|
189 if (song_info_timeout_source) |
6a9fdd5aee3a
Move timer updates out of the 100hz giant loop.
Daniel Drake <dsd@gentoo.org>
parents:
3001
diff
changeset
|
190 g_source_remove(song_info_timeout_source); |
6a9fdd5aee3a
Move timer updates out of the 100hz giant loop.
Daniel Drake <dsd@gentoo.org>
parents:
3001
diff
changeset
|
191 |
3082
91df04ec5ef7
From: Daniel Drake <dsd@gentoo.org>
William Pitcock <nenolod@atheme-project.org>
parents:
3042
diff
changeset
|
192 vis_playback_stop(); |
91df04ec5ef7
From: Daniel Drake <dsd@gentoo.org>
William Pitcock <nenolod@atheme-project.org>
parents:
3042
diff
changeset
|
193 |
2313 | 194 g_return_if_fail(mainwin_playstatus != NULL); |
195 playstatus_set_status_buffering(mainwin_playstatus, FALSE); | |
196 } | |
197 | |
198 static void | |
199 run_no_output_plugin_dialog(void) | |
200 { | |
201 const gchar *markup = | |
202 N_("<b><big>No output plugin selected.</big></b>\n" | |
203 "You have not selected an output plugin."); | |
204 | |
205 GtkWidget *dialog = | |
206 gtk_message_dialog_new_with_markup(GTK_WINDOW(mainwin), | |
207 GTK_DIALOG_DESTROY_WITH_PARENT, | |
208 GTK_MESSAGE_ERROR, | |
209 GTK_BUTTONS_OK, | |
210 _(markup)); | |
211 gtk_dialog_run(GTK_DIALOG(dialog)); | |
212 gtk_widget_destroy(dialog); | |
213 } | |
214 | |
215 gboolean | |
216 playback_play_file(PlaylistEntry *entry) | |
217 { | |
2436
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
218 InputPlayback * playback; |
2313 | 219 g_return_val_if_fail(entry != NULL, FALSE); |
220 | |
221 if (!get_current_output_plugin()) { | |
222 run_no_output_plugin_dialog(); | |
223 mainwin_stop_pushed(); | |
224 return FALSE; | |
225 } | |
226 | |
227 if (cfg.random_skin_on_play) | |
2443
93686e8815a4
[svn] - Changed playback_set_random_skin() to skin_set_random_skin()
mf0102
parents:
2438
diff
changeset
|
228 skin_set_random_skin(); |
2313 | 229 |
230 /* | |
231 * This is slightly uglier than the original version, but should | |
232 * fix the "crash" issues as seen in 0.2 when dealing with this situation. | |
233 * - nenolod | |
234 */ | |
235 if (!entry->decoder && | |
2420 | 236 (((entry->decoder = input_check_file(entry->filename, FALSE)) == NULL) || |
2313 | 237 !input_is_enabled(entry->decoder->filename))) |
238 { | |
2436
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
239 set_current_input_playback(NULL); |
2313 | 240 mainwin_set_info_text(); |
241 | |
242 return FALSE; | |
243 } | |
244 | |
2436
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
245 playback = g_new0(InputPlayback, 1); |
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
246 |
2313 | 247 entry->decoder->output = &psuedo_output_plugin; |
2436
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
248 |
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
249 playback->plugin = entry->decoder; |
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
250 playback->output = &psuedo_output_plugin; |
2438
8750a62abed8
[svn] Provide flags in InputPlayback for common plugin flag needs, and provide a
iabervon
parents:
2436
diff
changeset
|
251 playback->filename = g_strdup(entry->filename); |
2436
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
252 |
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
253 set_current_input_playback(playback); |
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
254 |
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
255 entry->decoder->play_file(playback); |
2313 | 256 |
257 ip_data.playing = TRUE; | |
258 | |
259 return TRUE; | |
260 } | |
261 | |
262 gboolean | |
263 playback_get_playing(void) | |
264 { | |
265 return ip_data.playing; | |
266 } | |
267 | |
268 gboolean | |
269 playback_get_paused(void) | |
270 { | |
271 return ip_data.paused; | |
272 } | |
273 | |
274 void | |
275 playback_seek(gint time) | |
276 { | |
277 gboolean restore_pause = FALSE; | |
278 gint l=0, r=0; | |
279 | |
2420 | 280 g_return_if_fail(ip_data.playing); |
2436
f346d30bf5ab
[svn] Change the input plugin API to use a struct for the currently-playing file.
iabervon
parents:
2420
diff
changeset
|
281 g_return_if_fail(get_current_input_playback()); |
2313 | 282 |
283 /* FIXME WORKAROUND...that should work with all plugins | |
284 * mute the volume, start playback again, do the seek, then pause again | |
285 * -Patrick Sudowe | |
286 */ | |
2420 | 287 if (ip_data.paused) |
2313 | 288 { |
2420 | 289 restore_pause = TRUE; |
290 output_get_volume(&l, &r); | |
291 output_set_volume(0,0); | |
292 playback_pause(); | |
2313 | 293 } |
294 | |
295 free_vis_data(); | |
2454 | 296 get_current_input_playback()->plugin->seek(get_current_input_playback(), time); |
2313 | 297 |
2420 | 298 if (restore_pause) |
2313 | 299 { |
2420 | 300 playback_pause(); |
301 output_set_volume(l, r); | |
2313 | 302 } |
303 } | |
304 | |
305 void | |
306 playback_seek_relative(gint offset) | |
307 { | |
308 gint time = CLAMP(playback_get_time() / 1000 + offset, | |
309 0, playlist_get_current_length(playlist_get_active()) / 1000 - 1); | |
310 playback_seek(time); | |
311 } |