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
|
|
40 #include "libaudacious/util.h"
|
|
41 #include "libaudacious/configdb.h"
|
|
42
|
|
43 #include "input.h"
|
|
44 #include "main.h"
|
|
45 #include "ui_main.h"
|
|
46 #include "ui_equalizer.h"
|
|
47 #include "output.h"
|
|
48 #include "playlist.h"
|
|
49 #include "ui_playlist.h"
|
|
50 #include "ui_skinselector.h"
|
|
51 #include "libaudacious/urldecode.h"
|
|
52 #include "util.h"
|
|
53
|
|
54
|
|
55 #include "playback.h"
|
|
56
|
|
57
|
|
58 /* FIXME: yuck!! this shouldn't be here... */
|
|
59 void
|
|
60 playback_set_random_skin(void)
|
|
61 {
|
|
62 SkinNode *node;
|
|
63 guint32 randval;
|
|
64
|
|
65 /* Get a random value to select the skin to use */
|
|
66 randval = g_random_int_range(0, g_list_length(skinlist));
|
|
67 node = g_list_nth(skinlist, randval)->data;
|
|
68 bmp_active_skin_load(node->path);
|
|
69 }
|
|
70
|
|
71 gint
|
|
72 playback_get_time(void)
|
|
73 {
|
|
74 if (!playback_get_playing())
|
|
75 return -1;
|
|
76
|
|
77 if (!get_current_input_plugin())
|
|
78 return -1;
|
|
79
|
|
80 return get_current_input_plugin()->get_time();
|
|
81 }
|
|
82
|
|
83 void
|
|
84 playback_initiate(void)
|
|
85 {
|
|
86 PlaylistEntry *entry = NULL;
|
|
87 Playlist *playlist = playlist_get_active();
|
|
88 gint penalty = 0;
|
|
89
|
|
90 if (playlist_get_length(playlist) == 0)
|
|
91 return;
|
|
92
|
|
93 if (playback_get_playing())
|
|
94 playback_stop();
|
|
95
|
|
96 vis_clear_data(mainwin_vis);
|
|
97 svis_clear_data(mainwin_svis);
|
|
98 mainwin_disable_seekbar();
|
|
99
|
|
100 entry = playlist_get_entry_to_play(playlist);
|
|
101
|
|
102 if (entry == NULL)
|
|
103 return;
|
|
104
|
|
105 /*
|
|
106 * If the playlist entry cannot be played, try to pick another one.
|
|
107 * If that does not work, e.g. entry == NULL, then bail.
|
|
108 *
|
|
109 * - nenolod
|
|
110 */
|
|
111 for ( penalty=0 ; penalty <= 15 && entry != NULL && !playback_play_file(entry) ; penalty++ )
|
|
112 {
|
|
113 playlist_next(playlist);
|
|
114
|
|
115 entry = playlist_get_entry_to_play(playlist);
|
|
116
|
|
117 /* XXX ew. workaround for a stupid bug where audacious will keep
|
|
118 * trying to play a file with no valid decoder.
|
|
119 */
|
|
120 if (entry == NULL)
|
|
121 return;
|
|
122
|
|
123 if (entry->decoder == NULL )
|
|
124 entry->decoder = input_check_file(entry->filename, FALSE);
|
|
125
|
|
126 /* if we hit 15 entries in a row with no valid decoder, just
|
|
127 * bail due to confusion
|
|
128 */
|
|
129 if (penalty == 15)
|
|
130 return;
|
|
131 }
|
|
132
|
|
133 if (playback_get_time() != -1) {
|
|
134 equalizerwin_load_auto_preset(entry->filename);
|
|
135 input_set_eq(cfg.equalizer_active, cfg.equalizer_preamp,
|
|
136 cfg.equalizer_bands);
|
|
137 output_set_eq(cfg.equalizer_active, cfg.equalizer_preamp,
|
|
138 cfg.equalizer_bands);
|
|
139 }
|
|
140
|
|
141 playlist_check_pos_current(playlist);
|
|
142 mainwin_set_info_text();
|
|
143 }
|
|
144
|
|
145 void
|
|
146 playback_pause(void)
|
|
147 {
|
|
148 if (!playback_get_playing())
|
|
149 return;
|
|
150
|
|
151 if (!get_current_input_plugin())
|
|
152 return;
|
|
153
|
|
154 ip_data.paused = !ip_data.paused;
|
|
155
|
|
156 if (get_current_input_plugin()->pause)
|
|
157 get_current_input_plugin()->pause(ip_data.paused);
|
|
158
|
|
159 g_return_if_fail(mainwin_playstatus != NULL);
|
|
160
|
|
161 if (ip_data.paused)
|
|
162 playstatus_set_status(mainwin_playstatus, STATUS_PAUSE);
|
|
163 else
|
|
164 playstatus_set_status(mainwin_playstatus, STATUS_PLAY);
|
|
165 }
|
|
166
|
|
167 void
|
|
168 playback_stop(void)
|
|
169 {
|
|
170 if (ip_data.playing && get_current_input_plugin()) {
|
|
171
|
|
172 if (playback_get_paused()) {
|
|
173 output_flush(get_written_time()); /* to avoid noise */
|
|
174 playback_pause();
|
|
175 }
|
|
176
|
|
177 ip_data.playing = FALSE;
|
|
178
|
|
179 if (get_current_input_plugin()->stop)
|
|
180 get_current_input_plugin()->stop();
|
|
181
|
|
182 free_vis_data();
|
|
183 ip_data.paused = FALSE;
|
|
184
|
|
185 if (input_info_text) {
|
|
186 g_free(input_info_text);
|
|
187 input_info_text = NULL;
|
|
188 mainwin_set_info_text();
|
|
189 }
|
|
190 }
|
|
191
|
|
192 ip_data.buffering = FALSE;
|
|
193 ip_data.playing = FALSE;
|
|
194
|
|
195 g_return_if_fail(mainwin_playstatus != NULL);
|
|
196 playstatus_set_status_buffering(mainwin_playstatus, FALSE);
|
|
197 }
|
|
198
|
|
199 static void
|
|
200 run_no_output_plugin_dialog(void)
|
|
201 {
|
|
202 const gchar *markup =
|
|
203 N_("<b><big>No output plugin selected.</big></b>\n"
|
|
204 "You have not selected an output plugin.");
|
|
205
|
|
206 GtkWidget *dialog =
|
|
207 gtk_message_dialog_new_with_markup(GTK_WINDOW(mainwin),
|
|
208 GTK_DIALOG_DESTROY_WITH_PARENT,
|
|
209 GTK_MESSAGE_ERROR,
|
|
210 GTK_BUTTONS_OK,
|
|
211 _(markup));
|
|
212 gtk_dialog_run(GTK_DIALOG(dialog));
|
|
213 gtk_widget_destroy(dialog);
|
|
214 }
|
|
215
|
|
216 gboolean
|
|
217 playback_play_file(PlaylistEntry *entry)
|
|
218 {
|
|
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)
|
|
228 playback_set_random_skin();
|
|
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 &&
|
|
236 (((entry->decoder = input_check_file(entry->filename, FALSE)) == NULL) ||
|
|
237 !input_is_enabled(entry->decoder->filename)))
|
|
238 {
|
|
239 input_file_not_playable(entry->filename);
|
|
240
|
|
241 set_current_input_plugin(NULL);
|
|
242 mainwin_set_info_text();
|
|
243
|
|
244 return FALSE;
|
|
245 }
|
|
246
|
|
247 set_current_input_plugin(entry->decoder);
|
|
248 entry->decoder->output = &psuedo_output_plugin;
|
|
249 entry->decoder->play_file(entry->filename);
|
|
250
|
|
251 ip_data.playing = TRUE;
|
|
252
|
|
253 return TRUE;
|
|
254 }
|
|
255
|
|
256 gboolean
|
|
257 playback_get_playing(void)
|
|
258 {
|
|
259 return ip_data.playing;
|
|
260 }
|
|
261
|
|
262 gboolean
|
|
263 playback_get_paused(void)
|
|
264 {
|
|
265 return ip_data.paused;
|
|
266 }
|
|
267
|
|
268 void
|
|
269 playback_seek(gint time)
|
|
270 {
|
|
271 gboolean restore_pause = FALSE;
|
|
272 gint l=0, r=0;
|
|
273
|
|
274 if (!ip_data.playing)
|
|
275 return;
|
|
276
|
|
277 if (!get_current_input_plugin())
|
|
278 return;
|
|
279
|
|
280 /* FIXME WORKAROUND...that should work with all plugins
|
|
281 * mute the volume, start playback again, do the seek, then pause again
|
|
282 * -Patrick Sudowe
|
|
283 */
|
|
284 if(ip_data.paused)
|
|
285 {
|
|
286 restore_pause = TRUE;
|
|
287 output_get_volume(&l, &r);
|
|
288 output_set_volume(0,0);
|
|
289 playback_pause();
|
|
290 }
|
|
291
|
|
292 free_vis_data();
|
|
293 get_current_input_plugin()->seek(time);
|
|
294
|
|
295 if(restore_pause)
|
|
296 {
|
|
297 playback_pause();
|
|
298 output_set_volume(l, r);
|
|
299 }
|
|
300 }
|
|
301
|
|
302 void
|
|
303 playback_seek_relative(gint offset)
|
|
304 {
|
|
305 gint time = CLAMP(playback_get_time() / 1000 + offset,
|
|
306 0, playlist_get_current_length(playlist_get_active()) / 1000 - 1);
|
|
307 playback_seek(time);
|
|
308 }
|