0
|
1 /* BMP - Cross-platform multimedia player
|
|
2 * Copyright (C) 2003-2004 BMP development team.
|
|
3 *
|
|
4 * Based on XMMS:
|
|
5 * Copyright (C) 1998-2003 XMMS development team.
|
|
6 *
|
|
7 * This program is free software; you can redistribute it and/or modify
|
|
8 * it under the terms of the GNU General Public License as published by
|
|
9 * the Free Software Foundation; either version 2 of the License, or
|
|
10 * (at your option) any later version.
|
|
11 *
|
|
12 * This program is distributed in the hope that it will be useful,
|
|
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
15 * GNU General Public License for more details.
|
|
16 *
|
|
17 * You should have received a copy of the GNU General Public License
|
|
18 * along with this program; if not, write to the Free Software
|
|
19 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
20 */
|
|
21
|
|
22 #ifdef HAVE_CONFIG_H
|
|
23 # include "config.h"
|
|
24 #endif
|
|
25
|
|
26
|
|
27 #include <glib.h>
|
|
28 #include <glib/gi18n.h>
|
|
29 #include <glib/gprintf.h>
|
|
30 #include <gtk/gtk.h>
|
|
31 #include <gdk/gdkx.h>
|
|
32 #include <gdk/gdkkeysyms.h>
|
|
33
|
|
34 #include <math.h>
|
|
35 #include <stdlib.h>
|
|
36 #include <string.h>
|
|
37
|
|
38 #include <X11/Xlib.h>
|
|
39
|
|
40 #include "textbox.h"
|
|
41 #include "mainwin.h"
|
|
42 #include "pixmaps.h"
|
|
43
|
|
44 #include "main.h"
|
|
45
|
|
46 #include "controlsocket.h"
|
|
47 #include "pluginenum.h"
|
|
48
|
|
49 #include "about.h"
|
|
50 #include "dnd.h"
|
|
51 #include "dock.h"
|
|
52 #include "equalizer.h"
|
|
53 #include "hints.h"
|
|
54 #include "input.h"
|
|
55 #include "playlistwin.h"
|
|
56 #include "prefswin.h"
|
|
57 #include "skinwin.h"
|
|
58
|
|
59 #include "hslider.h"
|
|
60 #include "menurow.h"
|
|
61 #include "monostereo.h"
|
|
62 #include "pbutton.h"
|
|
63 #include "playback.h"
|
|
64 #include "playlist.h"
|
|
65 #include "playlist_list.h"
|
|
66 #include "playstatus.h"
|
|
67 #include "sbutton.h"
|
|
68 #include "svis.h"
|
|
69 #include "textbox.h"
|
|
70 #include "urldecode.h"
|
|
71 #include "util.h"
|
|
72 #include "vis.h"
|
|
73 #include "visualization.h"
|
|
74 #include "libaudacious/configdb.h"
|
|
75
|
|
76
|
|
77 #define ITEM_SEPARATOR {"/-", NULL, NULL, 0, "<Separator>"}
|
|
78
|
|
79 /*
|
|
80 * If you change the menu above change these defines also
|
|
81 */
|
|
82
|
|
83 #define MAINWIN_VIS_MENU_VIS_MODE 1
|
|
84 #define MAINWIN_VIS_MENU_NUM_VIS_MODE 3
|
|
85 #define MAINWIN_VIS_MENU_ANALYZER_MODE 5
|
|
86 #define MAINWIN_VIS_MENU_NUM_ANALYZER_MODE 3
|
|
87 #define MAINWIN_VIS_MENU_ANALYZER_TYPE 9
|
|
88 #define MAINWIN_VIS_MENU_NUM_ANALYZER_TYPE 2
|
|
89 #define MAINWIN_VIS_MENU_ANALYZER_PEAKS 12
|
|
90 #define MAINWIN_VIS_MENU_SCOPE_MODE 14
|
|
91 #define MAINWIN_VIS_MENU_NUM_SCOPE_MODE 3
|
|
92 #define MAINWIN_VIS_MENU_WSHADEVU_MODE 18
|
|
93 #define MAINWIN_VIS_MENU_NUM_WSHADEVU_MODE 2
|
|
94 #define MAINWIN_VIS_MENU_REFRESH_RATE 21
|
|
95 #define MAINWIN_VIS_MENU_NUM_REFRESH_RATE 4
|
|
96 #define MAINWIN_VIS_MENU_AFALLOFF 26
|
|
97 #define MAINWIN_VIS_MENU_NUM_AFALLOFF 5
|
|
98 #define MAINWIN_VIS_MENU_PFALLOFF 32
|
|
99 #define MAINWIN_VIS_MENU_NUM_PFALLOFF 5
|
|
100
|
|
101 #define VOLSET_DISP_TIMES 5
|
|
102
|
|
103
|
|
104 enum {
|
|
105 MAINWIN_SONGNAME_FILEINFO,
|
|
106 MAINWIN_SONGNAME_JTF,
|
|
107 MAINWIN_SONGNAME_JTT,
|
|
108 MAINWIN_SONGNAME_SCROLL
|
|
109 };
|
|
110
|
|
111 enum {
|
|
112 MAINWIN_OPT_SKIN, MAINWIN_OPT_RELOADSKIN,
|
|
113 MAINWIN_OPT_REPEAT, MAINWIN_OPT_SHUFFLE, MAINWIN_OPT_NPA,
|
|
114 MAINWIN_OPT_TELAPSED, MAINWIN_OPT_TREMAINING,
|
|
115 MAINWIN_OPT_ALWAYS,
|
|
116 MAINWIN_OPT_STICKY,
|
|
117 MAINWIN_OPT_WS,
|
|
118 MAINWIN_OPT_PWS,
|
|
119 MAINWIN_OPT_EQWS
|
|
120 };
|
|
121
|
|
122 enum {
|
|
123 MAINWIN_VIS_ANALYZER, MAINWIN_VIS_SCOPE, MAINWIN_VIS_OFF,
|
|
124 MAINWIN_VIS_ANALYZER_NORMAL, MAINWIN_VIS_ANALYZER_FIRE,
|
|
125 MAINWIN_VIS_ANALYZER_VLINES,
|
|
126 MAINWIN_VIS_ANALYZER_LINES, MAINWIN_VIS_ANALYZER_BARS,
|
|
127 MAINWIN_VIS_ANALYZER_PEAKS,
|
|
128 MAINWIN_VIS_SCOPE_DOT, MAINWIN_VIS_SCOPE_LINE, MAINWIN_VIS_SCOPE_SOLID,
|
|
129 MAINWIN_VIS_VU_NORMAL, MAINWIN_VIS_VU_SMOOTH,
|
|
130 MAINWIN_VIS_REFRESH_FULL, MAINWIN_VIS_REFRESH_HALF,
|
|
131 MAINWIN_VIS_REFRESH_QUARTER, MAINWIN_VIS_REFRESH_EIGHTH,
|
|
132 MAINWIN_VIS_AFALLOFF_SLOWEST, MAINWIN_VIS_AFALLOFF_SLOW,
|
|
133 MAINWIN_VIS_AFALLOFF_MEDIUM, MAINWIN_VIS_AFALLOFF_FAST,
|
|
134 MAINWIN_VIS_AFALLOFF_FASTEST,
|
|
135 MAINWIN_VIS_PFALLOFF_SLOWEST, MAINWIN_VIS_PFALLOFF_SLOW,
|
|
136 MAINWIN_VIS_PFALLOFF_MEDIUM, MAINWIN_VIS_PFALLOFF_FAST,
|
|
137 MAINWIN_VIS_PFALLOFF_FASTEST,
|
|
138 MAINWIN_VIS_PLUGINS
|
|
139 };
|
|
140
|
|
141 enum {
|
|
142 MAINWIN_VIS_ACTIVE_MAINWIN, MAINWIN_VIS_ACTIVE_PLAYLISTWIN
|
|
143 };
|
|
144
|
|
145
|
|
146 typedef struct _PlaybackInfo PlaybackInfo;
|
|
147
|
|
148 struct _PlaybackInfo {
|
|
149 gchar *title;
|
|
150 gint bitrate;
|
|
151 gint frequency;
|
|
152 gint n_channels;
|
|
153 };
|
|
154
|
|
155
|
|
156 GtkWidget *mainwin = NULL;
|
|
157
|
|
158 static GdkBitmap *nullmask;
|
|
159 static gint balance;
|
|
160
|
|
161 GtkWidget *mainwin_jtf = NULL;
|
|
162 static GtkWidget *mainwin_jtt = NULL;
|
|
163
|
|
164 GtkItemFactory *mainwin_songname_menu, *mainwin_vis_menu;
|
|
165 GtkItemFactory *mainwin_general_menu, *mainwin_play_menu, *mainwin_add_menu;
|
|
166 GtkItemFactory *mainwin_view_menu;
|
|
167
|
|
168
|
|
169 GdkGC *mainwin_gc;
|
|
170 static GdkPixmap *mainwin_bg = NULL;
|
|
171
|
|
172 GtkAccelGroup *mainwin_accel = NULL;
|
|
173
|
|
174 static PButton *mainwin_menubtn;
|
|
175 static PButton *mainwin_minimize, *mainwin_shade, *mainwin_close;
|
|
176
|
|
177 static PButton *mainwin_rew, *mainwin_fwd;
|
|
178 static PButton *mainwin_eject;
|
|
179 static PButton *mainwin_play, *mainwin_pause, *mainwin_stop;
|
|
180
|
|
181 TButton *mainwin_shuffle, *mainwin_repeat, *mainwin_eq, *mainwin_pl;
|
|
182 TextBox *mainwin_info;
|
|
183
|
|
184 static TextBox *mainwin_rate_text, *mainwin_freq_text;
|
|
185 static TextBox *mainwin_stime_min, *mainwin_stime_sec;
|
|
186
|
|
187 PlayStatus *mainwin_playstatus;
|
|
188
|
|
189 static Number *mainwin_minus_num, *mainwin_10min_num, *mainwin_min_num;
|
|
190 static Number *mainwin_10sec_num, *mainwin_sec_num;
|
|
191
|
|
192 static gboolean setting_volume = FALSE;
|
|
193
|
|
194 Vis *active_vis;
|
|
195 Vis *mainwin_vis;
|
|
196 SVis *mainwin_svis;
|
|
197
|
|
198 static MenuRow *mainwin_menurow;
|
|
199 static HSlider *mainwin_volume, *mainwin_balance, *mainwin_position;
|
|
200 static HSlider *mainwin_sposition = NULL;
|
|
201 static MonoStereo *mainwin_monostereo;
|
|
202 static SButton *mainwin_srew, *mainwin_splay, *mainwin_spause;
|
|
203 static SButton *mainwin_sstop, *mainwin_sfwd, *mainwin_seject, *mainwin_about;
|
|
204
|
|
205 static GList *mainwin_wlist = NULL;
|
|
206
|
|
207 static gint mainwin_timeout_id;
|
|
208
|
|
209 G_LOCK_DEFINE_STATIC(mainwin_title);
|
|
210
|
|
211 static gboolean mainwin_force_redraw = FALSE;
|
|
212 static gchar *mainwin_title_text = NULL;
|
|
213 static gboolean mainwin_info_text_locked = FALSE;
|
|
214
|
|
215
|
|
216 static void mainwin_songname_menu_callback(gpointer user_data,
|
|
217 guint action,
|
|
218 GtkWidget * widget);
|
|
219
|
|
220 static void mainwin_vis_menu_callback(gpointer user_data,
|
|
221 guint action,
|
|
222 GtkWidget * widget);
|
|
223
|
|
224 static void mainwin_view_menu_callback(gpointer user_data,
|
|
225 guint action,
|
|
226 GtkWidget * widget);
|
|
227
|
|
228 static void mainwin_play_menu_callback(gpointer user_data,
|
|
229 guint action,
|
|
230 GtkWidget * widget);
|
|
231
|
|
232 /* Song name area menu */
|
|
233
|
|
234 static GtkItemFactoryEntry mainwin_songname_menu_entries[] = {
|
|
235 {N_("/View Track Details"), "<alt>i", mainwin_general_menu_callback,
|
|
236 MAINWIN_GENERAL_FILEINFO, "<ImageItem>", my_pixbuf},
|
|
237 {"/-", NULL, NULL, 0, "<Separator>"},
|
|
238 {N_("/Autoscroll Songname"), NULL, mainwin_songname_menu_callback,
|
|
239 MAINWIN_SONGNAME_SCROLL, "<ToggleItem>"},
|
|
240 };
|
|
241
|
|
242 static gint mainwin_songname_menu_entries_num =
|
|
243 G_N_ELEMENTS(mainwin_songname_menu_entries);
|
|
244
|
|
245 /* Mini-visualizer area menu */
|
|
246
|
|
247 static GtkItemFactoryEntry mainwin_vis_menu_entries[] = {
|
|
248 {N_("/Visualization Mode"), NULL, NULL, 0, "<Branch>"},
|
|
249 {N_("/Visualization Mode/Analyzer"), NULL, mainwin_vis_menu_callback,
|
|
250 MAINWIN_VIS_ANALYZER, "<RadioItem>"},
|
|
251 {N_("/Visualization Mode/Scope"), NULL, mainwin_vis_menu_callback,
|
|
252 MAINWIN_VIS_SCOPE, "/Visualization Mode/Analyzer"},
|
|
253 {N_("/Visualization Mode/Off"), NULL, mainwin_vis_menu_callback,
|
|
254 MAINWIN_VIS_OFF, "/Visualization Mode/Analyzer"},
|
|
255 {N_("/Analyzer Mode"), NULL, NULL, 0, "<Branch>"},
|
|
256 {N_("/Analyzer Mode/Normal"), NULL, mainwin_vis_menu_callback,
|
|
257 MAINWIN_VIS_ANALYZER_NORMAL, "<RadioItem>"},
|
|
258 {N_("/Analyzer Mode/Fire"), NULL, mainwin_vis_menu_callback,
|
|
259 MAINWIN_VIS_ANALYZER_FIRE, "/Analyzer Mode/Normal"},
|
|
260 {N_("/Analyzer Mode/Vertical Lines"), NULL, mainwin_vis_menu_callback,
|
|
261 MAINWIN_VIS_ANALYZER_VLINES, "/Analyzer Mode/Normal"},
|
|
262 {"/Analyzer Mode/-", NULL, NULL, 0, "<Separator>"},
|
|
263 {N_("/Analyzer Mode/Lines"), NULL, mainwin_vis_menu_callback,
|
|
264 MAINWIN_VIS_ANALYZER_LINES, "<RadioItem>"},
|
|
265 {N_("/Analyzer Mode/Bars"), NULL, mainwin_vis_menu_callback,
|
|
266 MAINWIN_VIS_ANALYZER_BARS, "/Analyzer Mode/Lines"},
|
|
267 {"/Analyzer Mode/-", NULL, NULL, 0, "<Separator>"},
|
|
268 {N_("/Analyzer Mode/Peaks"), NULL, mainwin_vis_menu_callback,
|
|
269 MAINWIN_VIS_ANALYZER_PEAKS, "<ToggleItem>"},
|
|
270 {N_("/Scope Mode"), NULL, NULL, 0, "<Branch>"},
|
|
271 {N_("/Scope Mode/Dot Scope"), NULL, mainwin_vis_menu_callback,
|
|
272 MAINWIN_VIS_SCOPE_DOT, "<RadioItem>"},
|
|
273 {N_("/Scope Mode/Line Scope"), NULL, mainwin_vis_menu_callback,
|
|
274 MAINWIN_VIS_SCOPE_LINE, "/Scope Mode/Dot Scope"},
|
|
275 {N_("/Scope Mode/Solid Scope"), NULL, mainwin_vis_menu_callback,
|
|
276 MAINWIN_VIS_SCOPE_SOLID, "/Scope Mode/Dot Scope"},
|
|
277 {N_("/WindowShade VU Mode"), NULL, NULL, 0, "<Branch>"},
|
|
278 {N_("/WindowShade VU Mode/Normal"), NULL, mainwin_vis_menu_callback,
|
|
279 MAINWIN_VIS_VU_NORMAL, "<RadioItem>"},
|
|
280 {N_("/WindowShade VU Mode/Smooth"), NULL, mainwin_vis_menu_callback,
|
|
281 MAINWIN_VIS_VU_SMOOTH, "/WindowShade VU Mode/Normal"},
|
|
282 {N_("/Refresh Rate"), NULL, NULL, 0, "<Branch>"},
|
|
283 {N_("/Refresh Rate/Full (~50 fps)"), NULL, mainwin_vis_menu_callback,
|
|
284 MAINWIN_VIS_REFRESH_FULL, "<RadioItem>"},
|
|
285 {N_("/Refresh Rate/Half (~25 fps)"), NULL, mainwin_vis_menu_callback,
|
|
286 MAINWIN_VIS_REFRESH_HALF, "/Refresh Rate/Full (~50 fps)"},
|
|
287 {N_("/Refresh Rate/Quarter (~13 fps)"), NULL, mainwin_vis_menu_callback,
|
|
288 MAINWIN_VIS_REFRESH_QUARTER, "/Refresh Rate/Full (~50 fps)"},
|
|
289 {N_("/Refresh Rate/Eighth (~6 fps)"), NULL, mainwin_vis_menu_callback,
|
|
290 MAINWIN_VIS_REFRESH_EIGHTH, "/Refresh Rate/Full (~50 fps)"},
|
|
291 {N_("/Analyzer Falloff"), NULL, NULL, 0, "<Branch>"},
|
|
292 {N_("/Analyzer Falloff/Slowest"), NULL, mainwin_vis_menu_callback,
|
|
293 MAINWIN_VIS_AFALLOFF_SLOWEST, "<RadioItem>"},
|
|
294 {N_("/Analyzer Falloff/Slow"), NULL, mainwin_vis_menu_callback,
|
|
295 MAINWIN_VIS_AFALLOFF_SLOW, "/Analyzer Falloff/Slowest"},
|
|
296 {N_("/Analyzer Falloff/Medium"), NULL, mainwin_vis_menu_callback,
|
|
297 MAINWIN_VIS_AFALLOFF_MEDIUM, "/Analyzer Falloff/Slowest"},
|
|
298 {N_("/Analyzer Falloff/Fast"), NULL, mainwin_vis_menu_callback,
|
|
299 MAINWIN_VIS_AFALLOFF_FAST, "/Analyzer Falloff/Slowest"},
|
|
300 {N_("/Analyzer Falloff/Fastest"), NULL, mainwin_vis_menu_callback,
|
|
301 MAINWIN_VIS_AFALLOFF_FASTEST, "/Analyzer Falloff/Slowest"},
|
|
302 {N_("/Peaks Falloff"), NULL, NULL, 0, "<Branch>"},
|
|
303 {N_("/Peaks Falloff/Slowest"), NULL, mainwin_vis_menu_callback,
|
|
304 MAINWIN_VIS_PFALLOFF_SLOWEST, "<RadioItem>"},
|
|
305 {N_("/Peaks Falloff/Slow"), NULL, mainwin_vis_menu_callback,
|
|
306 MAINWIN_VIS_PFALLOFF_SLOW, "/Peaks Falloff/Slowest"},
|
|
307 {N_("/Peaks Falloff/Medium"), NULL, mainwin_vis_menu_callback,
|
|
308 MAINWIN_VIS_PFALLOFF_MEDIUM, "/Peaks Falloff/Slowest"},
|
|
309 {N_("/Peaks Falloff/Fast"), NULL, mainwin_vis_menu_callback,
|
|
310 MAINWIN_VIS_PFALLOFF_FAST, "/Peaks Falloff/Slowest"},
|
|
311 {N_("/Peaks Falloff/Fastest"), NULL, mainwin_vis_menu_callback,
|
|
312 MAINWIN_VIS_PFALLOFF_FASTEST, "/Peaks Falloff/Slowest"}
|
|
313 };
|
|
314
|
|
315 static const gint mainwin_vis_menu_entries_num =
|
|
316 G_N_ELEMENTS(mainwin_vis_menu_entries);
|
|
317
|
|
318 /* Playback menu (now used only for accelerators) */
|
|
319
|
|
320 static GtkItemFactoryEntry mainwin_playback_menu_entries[] = {
|
|
321 /*
|
|
322 {N_("/Play CD"), "<alt>C", mainwin_general_menu_callback,
|
|
323 MAINWIN_GENERAL_PLAYCD, "<StockItem>", GTK_STOCK_CDROM},
|
|
324 {"/-", NULL, NULL, 0, "<Separator>"},
|
|
325 */
|
|
326 {N_("/Repeat"), "R", mainwin_play_menu_callback,
|
|
327 MAINWIN_OPT_REPEAT, "<ToggleItem>"},
|
|
328 {N_("/Shuffle"), "S", mainwin_play_menu_callback,
|
|
329 MAINWIN_OPT_SHUFFLE, "<ToggleItem>"},
|
|
330 /*
|
|
331 {N_("/No Playlist Advance"), "<control>N", mainwin_play_menu_callback,
|
|
332 MAINWIN_OPT_NPA, "<ToggleItem>"},
|
|
333 */
|
|
334 {"/-", NULL, NULL, 0, "<Separator>"},
|
|
335 {N_("/Play"), "x", mainwin_general_menu_callback,
|
|
336 MAINWIN_GENERAL_PLAY, "<Item>"},
|
|
337 {N_("/Pause"), "c", mainwin_general_menu_callback,
|
|
338 MAINWIN_GENERAL_PAUSE, "<Item>"},
|
|
339 {N_("/Stop"), "v", mainwin_general_menu_callback,
|
|
340 MAINWIN_GENERAL_STOP, "<StockItem>", GTK_STOCK_STOP},
|
|
341 {N_("/Previous"), "z", mainwin_general_menu_callback,
|
|
342 MAINWIN_GENERAL_PREV, "<StockItem>", GTK_STOCK_GO_BACK},
|
|
343 {N_("/Next"), "b", mainwin_general_menu_callback,
|
|
344 MAINWIN_GENERAL_NEXT, "<StockItem>", GTK_STOCK_GO_FORWARD},
|
|
345 {"/-", NULL, NULL, 0, "<Separator>"},
|
|
346 {N_("/Jump to Playlist Start"), "<control>Z", mainwin_general_menu_callback,
|
|
347 MAINWIN_GENERAL_START, "<StockItem>", GTK_STOCK_GOTO_TOP},
|
|
348 {"/-", NULL, NULL, 0, "<Separator>"},
|
|
349 {N_("/Jump to File"), "J", mainwin_general_menu_callback,
|
|
350 MAINWIN_GENERAL_JTF, "<StockItem>", GTK_STOCK_JUMP_TO},
|
|
351 {N_("/Jump to Time"), "<control>J", mainwin_general_menu_callback,
|
|
352 MAINWIN_GENERAL_JTT, "<StockItem>", GTK_STOCK_JUMP_TO},
|
|
353 };
|
|
354
|
|
355 static const gint mainwin_playback_menu_entries_num =
|
|
356 G_N_ELEMENTS(mainwin_playback_menu_entries);
|
|
357
|
|
358 /* Main menu */
|
|
359
|
|
360 static GtkItemFactoryEntry mainwin_general_menu_entries[] = {
|
|
361 { N_("/View Track Details"), "<alt>I", mainwin_general_menu_callback,
|
|
362 MAINWIN_GENERAL_FILEINFO, "<ImageItem>", my_pixbuf},
|
|
363 {"/-", NULL, NULL, 0, "<Separator>"},
|
|
364 {N_("/Preferences"), "<control>P", mainwin_general_menu_callback,
|
|
365 MAINWIN_GENERAL_PREFS, "<StockItem>", GTK_STOCK_PREFERENCES},
|
|
366 {N_("/_View"), NULL, NULL, 0, "<Item>"},
|
|
367 {"/-", NULL, NULL, 0, "<Separator>"},
|
|
368 {N_("/About Audacious"), NULL, mainwin_general_menu_callback,
|
|
369 MAINWIN_GENERAL_ABOUT, "<StockItem>", GTK_STOCK_DIALOG_INFO},
|
|
370 {N_("/_Quit"), NULL, mainwin_general_menu_callback,
|
|
371 MAINWIN_GENERAL_EXIT, "<StockItem>", GTK_STOCK_QUIT}
|
|
372 };
|
|
373
|
|
374 static const gint mainwin_general_menu_entries_num =
|
|
375 G_N_ELEMENTS(mainwin_general_menu_entries);
|
|
376
|
|
377 /* Add submenu */
|
|
378
|
|
379 static GtkItemFactoryEntry mainwin_add_menu_entries[] = {
|
|
380 {N_("/Files..."), "f", mainwin_general_menu_callback,
|
|
381 MAINWIN_GENERAL_PLAYFILE, "<StockItem>", GTK_STOCK_OPEN},
|
|
382 /*
|
|
383 {N_("/Folders..."), "d",
|
|
384 mainwin_general_menu_callback,
|
|
385 MAINWIN_GENERAL_PLAYDIRECTORY, "<StockItem>", GTK_STOCK_OPEN},
|
|
386 */
|
|
387 {N_("/Internet location..."), "<control>h", mainwin_general_menu_callback,
|
|
388 MAINWIN_GENERAL_PLAYLOCATION, "<StockItem>", GTK_STOCK_NETWORK},
|
|
389 };
|
|
390
|
|
391 static const gint mainwin_add_menu_entries_num =
|
|
392 G_N_ELEMENTS(mainwin_add_menu_entries);
|
|
393
|
|
394
|
|
395 /*
|
|
396 */
|
|
397
|
|
398 /* View submenu */
|
|
399
|
|
400 static GtkItemFactoryEntry mainwin_view_menu_entries[] = {
|
|
401 {N_("/Show Playlist Editor"), "<alt>E", mainwin_general_menu_callback,
|
|
402 MAINWIN_GENERAL_SHOWPLWIN, "<ToggleItem>"},
|
|
403 {N_("/Show Equalizer"), "<alt>G", mainwin_general_menu_callback,
|
|
404 MAINWIN_GENERAL_SHOWEQWIN, "<ToggleItem>"},
|
|
405 {"/-", NULL, NULL, 0, "<Separator>"},
|
|
406 {N_("/Time Elapsed"), "<control>E", mainwin_view_menu_callback,
|
|
407 MAINWIN_OPT_TELAPSED, "<RadioItem>"},
|
|
408 {N_("/Time Remaining"), "<control>R", mainwin_view_menu_callback,
|
|
409 MAINWIN_OPT_TREMAINING, "/Time Elapsed"},
|
|
410 {"/-", NULL, NULL, 0, "<Separator>"},
|
|
411 {N_("/Always On Top"), "<control>o", mainwin_view_menu_callback,
|
|
412 MAINWIN_OPT_ALWAYS, "<ToggleItem>"},
|
|
413 {N_("/Put on All Workspaces"), "<control>S",
|
|
414 mainwin_view_menu_callback, MAINWIN_OPT_STICKY, "<ToggleItem>"},
|
|
415 {"/-", NULL, NULL, 0, "<Separator>"},
|
|
416 {N_("/Roll up Player"), "<control>W", mainwin_view_menu_callback,
|
|
417 MAINWIN_OPT_WS, "<ToggleItem>"},
|
|
418 {N_("/Roll up Playlist Editor"), "<control><shift>W", mainwin_view_menu_callback,
|
|
419 MAINWIN_OPT_PWS, "<ToggleItem>"},
|
|
420 {N_("/Roll up Equalizer"), "<control><alt>W", mainwin_view_menu_callback,
|
|
421 MAINWIN_OPT_EQWS, "<ToggleItem>"}
|
|
422 };
|
|
423
|
|
424 static const gint mainwin_view_menu_entries_num =
|
|
425 G_N_ELEMENTS(mainwin_view_menu_entries);
|
|
426
|
|
427
|
|
428 static PlaybackInfo playback_info = { NULL, 0, 0, 0 };
|
|
429
|
|
430
|
|
431 static gint mainwin_idle_func(gpointer data);
|
|
432
|
|
433 static void set_timer_mode_menu_cb(TimerMode mode);
|
|
434 static void set_timer_mode(TimerMode mode);
|
|
435
|
|
436
|
|
437 /* FIXME: placed here for now */
|
|
438 void
|
|
439 playback_get_sample_params(gint * bitrate,
|
|
440 gint * frequency,
|
|
441 gint * n_channels)
|
|
442 {
|
|
443 if (bitrate)
|
|
444 *bitrate = playback_info.bitrate;
|
|
445
|
|
446 if (frequency)
|
|
447 *frequency = playback_info.frequency;
|
|
448
|
|
449 if (n_channels)
|
|
450 *n_channels = playback_info.n_channels;
|
|
451 }
|
|
452
|
|
453 static void
|
|
454 playback_set_sample_params(gint bitrate,
|
|
455 gint frequency,
|
|
456 gint n_channels)
|
|
457 {
|
|
458 if (bitrate >= 0)
|
|
459 playback_info.bitrate = bitrate;
|
|
460
|
|
461 if (frequency >= 0)
|
|
462 playback_info.frequency = frequency;
|
|
463
|
|
464 if (n_channels >= 0)
|
|
465 playback_info.n_channels = n_channels;
|
|
466 }
|
|
467
|
|
468 static void
|
|
469 mainwin_set_title_scroll(gboolean scroll)
|
|
470 {
|
|
471 cfg.autoscroll = scroll;
|
|
472 textbox_set_scroll(mainwin_info, cfg.autoscroll);
|
|
473 }
|
|
474
|
|
475
|
|
476 void
|
|
477 mainwin_set_always_on_top(gboolean always)
|
|
478 {
|
|
479 GtkWidget *widget = gtk_item_factory_get_widget(mainwin_view_menu,
|
|
480 "/Always On Top");
|
|
481 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(widget),
|
|
482 mainwin_menurow->mr_always_selected);
|
|
483 }
|
|
484
|
|
485 static void
|
|
486 mainwin_set_shape_mask(void)
|
|
487 {
|
|
488 GdkBitmap *mask;
|
|
489
|
|
490 if (!cfg.player_visible)
|
|
491 return;
|
|
492
|
|
493 mask = skin_get_mask(bmp_active_skin, SKIN_MASK_MAIN + cfg.player_shaded);
|
|
494 gtk_widget_shape_combine_mask(mainwin, mask, 0, 0);
|
|
495 }
|
|
496
|
|
497 static void
|
|
498 mainwin_set_shade(gboolean shaded)
|
|
499 {
|
|
500 GtkWidget *widget;
|
|
501 widget = gtk_item_factory_get_widget(mainwin_view_menu,
|
|
502 "/Roll up Player");
|
|
503 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(widget), shaded);
|
|
504 }
|
|
505
|
|
506 static void
|
|
507 mainwin_set_shade_menu_cb(gboolean shaded)
|
|
508 {
|
|
509 cfg.player_shaded = shaded;
|
|
510
|
|
511 mainwin_set_shape_mask();
|
|
512
|
|
513 if (shaded) {
|
|
514 dock_shade(dock_window_list, GTK_WINDOW(mainwin),
|
|
515 MAINWIN_SHADED_HEIGHT);
|
|
516
|
|
517 widget_show(WIDGET(mainwin_svis));
|
|
518 vis_clear_data(mainwin_vis);
|
|
519
|
|
520 widget_show(WIDGET(mainwin_srew));
|
|
521 widget_show(WIDGET(mainwin_splay));
|
|
522 widget_show(WIDGET(mainwin_spause));
|
|
523 widget_show(WIDGET(mainwin_sstop));
|
|
524 widget_show(WIDGET(mainwin_sfwd));
|
|
525 widget_show(WIDGET(mainwin_seject));
|
|
526
|
|
527 widget_show(WIDGET(mainwin_stime_min));
|
|
528 widget_show(WIDGET(mainwin_stime_sec));
|
|
529
|
|
530 textbox_set_scroll(mainwin_info, FALSE);
|
|
531 if (bmp_playback_get_playing()
|
|
532 && playlist_get_current_length() != -1)
|
|
533 widget_show(WIDGET(mainwin_sposition));
|
|
534
|
|
535 mainwin_shade->pb_ny = mainwin_shade->pb_py = 27;
|
|
536 }
|
|
537 else {
|
|
538 dock_shade(dock_window_list, GTK_WINDOW(mainwin), MAINWIN_HEIGHT);
|
|
539
|
|
540 widget_hide(WIDGET(mainwin_svis));
|
|
541 svis_clear_data(mainwin_svis);
|
|
542
|
|
543 widget_hide(WIDGET(mainwin_srew));
|
|
544 widget_hide(WIDGET(mainwin_splay));
|
|
545 widget_hide(WIDGET(mainwin_spause));
|
|
546 widget_hide(WIDGET(mainwin_sstop));
|
|
547 widget_hide(WIDGET(mainwin_sfwd));
|
|
548 widget_hide(WIDGET(mainwin_seject));
|
|
549
|
|
550 widget_hide(WIDGET(mainwin_stime_min));
|
|
551 widget_hide(WIDGET(mainwin_stime_sec));
|
|
552 widget_hide(WIDGET(mainwin_sposition));
|
|
553
|
|
554 textbox_set_scroll(mainwin_info, TRUE);
|
|
555 mainwin_shade->pb_ny = mainwin_shade->pb_py = 18;
|
|
556 }
|
|
557
|
|
558 draw_main_window(TRUE);
|
|
559 }
|
|
560
|
|
561 static void
|
|
562 mainwin_vis_set_active_vis(gint new_vis)
|
|
563 {
|
|
564 switch (new_vis) {
|
|
565 case MAINWIN_VIS_ACTIVE_MAINWIN:
|
|
566 playlistwin_vis_disable();
|
|
567 active_vis = mainwin_vis;
|
|
568 break;
|
|
569 case MAINWIN_VIS_ACTIVE_PLAYLISTWIN:
|
|
570 playlistwin_vis_enable();
|
|
571 active_vis = playlistwin_vis;
|
|
572 break;
|
|
573 }
|
|
574 }
|
|
575
|
|
576 static void
|
|
577 mainwin_vis_set_refresh(RefreshRate rate)
|
|
578 {
|
|
579 cfg.vis_refresh = rate;
|
|
580 }
|
|
581
|
|
582 static void
|
|
583 mainwin_vis_set_afalloff(FalloffSpeed speed)
|
|
584 {
|
|
585 cfg.analyzer_falloff = speed;
|
|
586 }
|
|
587
|
|
588 static void
|
|
589 mainwin_vis_set_pfalloff(FalloffSpeed speed)
|
|
590 {
|
|
591 cfg.peaks_falloff = speed;
|
|
592 }
|
|
593
|
|
594 static void
|
|
595 mainwin_vis_set_analyzer_mode(AnalyzerMode mode)
|
|
596 {
|
|
597 cfg.analyzer_mode = mode;
|
|
598 }
|
|
599
|
|
600 static void
|
|
601 mainwin_vis_set_analyzer_type(AnalyzerType mode)
|
|
602 {
|
|
603 cfg.analyzer_type = mode;
|
|
604 }
|
|
605
|
|
606 void
|
|
607 mainwin_vis_set_type(VisType mode)
|
|
608 {
|
|
609 gchar *path =
|
|
610 mainwin_vis_menu_entries[MAINWIN_VIS_MENU_VIS_MODE + mode].path;
|
|
611 GtkWidget *widget = gtk_item_factory_get_widget(mainwin_vis_menu, path);
|
|
612 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(widget), TRUE);
|
|
613 }
|
|
614
|
|
615 static void
|
|
616 mainwin_vis_set_type_menu_cb(VisType mode)
|
|
617 {
|
|
618 cfg.vis_type = mode;
|
|
619
|
|
620 if (mode == VIS_OFF) {
|
|
621 if (cfg.player_shaded && cfg.player_visible)
|
|
622 svis_clear(mainwin_svis);
|
|
623 else
|
|
624 vis_clear(active_vis);
|
|
625 }
|
|
626 if (mode == VIS_ANALYZER) {
|
|
627 vis_clear_data(active_vis);
|
|
628 svis_clear_data(mainwin_svis);
|
|
629 }
|
|
630 }
|
|
631
|
|
632 static void
|
|
633 mainwin_menubtn_cb(void)
|
|
634 {
|
|
635 gint x, y;
|
|
636 gtk_window_get_position(GTK_WINDOW(mainwin), &x, &y);
|
|
637 util_item_factory_popup(mainwin_general_menu,
|
|
638 x + 6,
|
|
639 y + MAINWIN_SHADED_HEIGHT,
|
|
640 1, GDK_CURRENT_TIME);
|
|
641 }
|
|
642
|
|
643 void
|
|
644 mainwin_minimize_cb(void)
|
|
645 {
|
|
646 if (!mainwin)
|
|
647 return;
|
|
648
|
|
649 gtk_window_iconify(GTK_WINDOW(mainwin));
|
|
650 }
|
|
651
|
|
652 static void
|
|
653 mainwin_shade_toggle(void)
|
|
654 {
|
|
655 mainwin_set_shade(!cfg.player_shaded);
|
|
656 }
|
|
657
|
|
658 void
|
|
659 mainwin_quit_cb(void)
|
|
660 {
|
|
661 gtk_widget_hide(equalizerwin);
|
|
662 gtk_widget_hide(playlistwin);
|
|
663 gtk_widget_hide(mainwin);
|
|
664 gdk_flush();
|
|
665
|
|
666 g_source_remove(mainwin_timeout_id);
|
|
667
|
|
668 util_set_cursor(NULL);
|
|
669
|
|
670 bmp_config_save();
|
|
671 gtk_accel_map_save(bmp_paths[BMP_PATH_ACCEL_FILE]);
|
|
672
|
|
673 ctrlsocket_cleanup();
|
|
674
|
|
675 playlist_stop_get_info_thread();
|
|
676 playlist_clear();
|
|
677
|
|
678 plugin_system_cleanup();
|
|
679
|
|
680 gtk_main_quit();
|
|
681 }
|
|
682
|
|
683 static void
|
|
684 mainwin_destroy(GtkWidget * widget, gpointer data)
|
|
685 {
|
|
686 mainwin_quit_cb();
|
|
687 }
|
|
688
|
|
689 static void
|
|
690 mainwin_draw_titlebar(gboolean focus)
|
|
691 {
|
|
692 skin_draw_mainwin_titlebar(bmp_active_skin, mainwin_bg, mainwin_gc,
|
|
693 cfg.player_shaded, focus || !cfg.dim_titlebar);
|
|
694 }
|
|
695
|
|
696 void
|
|
697 draw_main_window(gboolean force)
|
|
698 {
|
|
699 GList *wl;
|
|
700 Widget *w;
|
|
701 gboolean redraw;
|
|
702
|
|
703 if (!cfg.player_visible)
|
|
704 return;
|
|
705
|
|
706 widget_list_lock(mainwin_wlist);
|
|
707
|
|
708 if (force) {
|
|
709 if (!cfg.player_shaded)
|
|
710 skin_draw_pixmap(bmp_active_skin, mainwin_bg, mainwin_gc,
|
|
711 SKIN_MAIN, 0, 0, 0, 0, MAINWIN_WIDTH,
|
|
712 MAINWIN_HEIGHT);
|
|
713 mainwin_draw_titlebar(gtk_window_has_toplevel_focus
|
|
714 (GTK_WINDOW(mainwin)));
|
|
715 }
|
|
716
|
|
717 widget_list_draw(mainwin_wlist, &redraw, force);
|
|
718
|
|
719 if (redraw || force) {
|
|
720 if (force) {
|
|
721 gdk_window_clear(mainwin->window);
|
|
722 }
|
|
723 else {
|
|
724 for (wl = mainwin_wlist; wl; wl = g_list_next(wl)) {
|
|
725 w = WIDGET(wl->data);
|
|
726
|
|
727 if (!w->redraw || !w->visible)
|
|
728 continue;
|
|
729
|
|
730 gdk_window_clear_area(mainwin->window, w->x, w->y,
|
|
731 w->width, w->height);
|
|
732 w->redraw = FALSE;
|
|
733 }
|
|
734 }
|
|
735
|
|
736 gdk_flush();
|
|
737 }
|
|
738
|
|
739 widget_list_unlock(mainwin_wlist);
|
|
740 }
|
|
741
|
|
742
|
|
743 void
|
|
744 mainwin_set_info_text(void)
|
|
745 {
|
|
746 gchar *text;
|
|
747
|
|
748 if (mainwin_info_text_locked)
|
|
749 return;
|
|
750
|
|
751 if ((text = input_get_info_text()) != NULL) {
|
|
752 textbox_set_text(mainwin_info, text);
|
|
753 g_free(text);
|
|
754 }
|
|
755 else if ((text = playlist_get_info_text()) != NULL) {
|
|
756 textbox_set_text(mainwin_info, text);
|
|
757 g_free(text);
|
|
758 }
|
|
759 }
|
|
760
|
|
761 void
|
|
762 mainwin_lock_info_text(const gchar * text)
|
|
763 {
|
|
764 mainwin_info_text_locked = TRUE;
|
|
765 textbox_set_text(mainwin_info, text);
|
|
766 }
|
|
767
|
|
768 void
|
|
769 mainwin_release_info_text(void)
|
|
770 {
|
|
771 mainwin_info_text_locked = FALSE;
|
|
772 mainwin_set_info_text();
|
|
773 }
|
|
774
|
|
775
|
|
776 static gchar *
|
|
777 make_mainwin_title(const gchar * title)
|
|
778 {
|
|
779 if (title)
|
|
780 return g_strdup_printf(_("%s - Audacious"), title);
|
|
781 else
|
|
782 return g_strdup(_("Audacious"));
|
|
783 }
|
|
784
|
|
785 void
|
|
786 mainwin_set_song_title(const gchar * title)
|
|
787 {
|
|
788 G_LOCK(mainwin_title);
|
|
789 g_free(mainwin_title_text);
|
|
790 mainwin_title_text = make_mainwin_title(title);
|
|
791 G_UNLOCK(mainwin_title);
|
|
792 }
|
|
793
|
|
794 void
|
|
795 mainwin_set_song_info(gint bitrate,
|
|
796 gint frequency,
|
|
797 gint n_channels)
|
|
798 {
|
|
799 gchar text[10];
|
|
800 gchar *title;
|
|
801
|
|
802 playback_set_sample_params(bitrate, frequency, n_channels);
|
|
803
|
|
804 if (bitrate != -1) {
|
|
805 bitrate /= 1000;
|
|
806
|
|
807 if (bitrate < 1000) {
|
|
808 /* Show bitrate in 1000s */
|
|
809 g_snprintf(text, sizeof(text), "%3d", bitrate);
|
|
810 textbox_set_text(mainwin_rate_text, text);
|
|
811 }
|
|
812 else {
|
|
813 /* Show bitrate in 100,000s */
|
|
814 bitrate /= 100;
|
|
815 g_snprintf(text, sizeof(text), "%2dH", bitrate);
|
|
816 textbox_set_text(mainwin_rate_text, text);
|
|
817 }
|
|
818 }
|
|
819 else
|
|
820 textbox_set_text(mainwin_rate_text, _("VBR"));
|
|
821
|
|
822 /* Show sampling frequency in kHz */
|
|
823 g_snprintf(text, sizeof(text), "%2d", frequency / 1000);
|
|
824 textbox_set_text(mainwin_freq_text, text);
|
|
825
|
|
826 monostereo_set_num_channels(mainwin_monostereo, n_channels);
|
|
827
|
|
828 widget_show(WIDGET(mainwin_minus_num));
|
|
829 widget_show(WIDGET(mainwin_10min_num));
|
|
830 widget_show(WIDGET(mainwin_min_num));
|
|
831 widget_show(WIDGET(mainwin_10sec_num));
|
|
832 widget_show(WIDGET(mainwin_sec_num));
|
|
833
|
|
834 if (!bmp_playback_get_paused())
|
|
835 playstatus_set_status(mainwin_playstatus, STATUS_PLAY);
|
|
836
|
|
837 if (playlist_get_current_length() != -1) {
|
|
838 if (cfg.player_shaded)
|
|
839 widget_show(WIDGET(mainwin_sposition));
|
|
840 widget_show(WIDGET(mainwin_position));
|
|
841 }
|
|
842 else {
|
|
843 widget_hide(WIDGET(mainwin_position));
|
|
844 widget_hide(WIDGET(mainwin_sposition));
|
|
845 mainwin_force_redraw = TRUE;
|
|
846 }
|
|
847
|
|
848 title = playlist_get_info_text();
|
|
849 mainwin_set_song_title(title);
|
|
850 g_free(title);
|
|
851 }
|
|
852
|
|
853 void
|
|
854 mainwin_clear_song_info(void)
|
|
855 {
|
|
856 /* clear title */
|
|
857 G_LOCK(mainwin_title);
|
|
858 g_free(mainwin_title_text);
|
|
859 mainwin_title_text = NULL;
|
|
860 G_UNLOCK(mainwin_title);
|
|
861
|
|
862 /* clear sampling parameters */
|
|
863 playback_set_sample_params(0, 0, 0);
|
|
864
|
|
865 mainwin_position->hs_pressed = FALSE;
|
|
866 mainwin_sposition->hs_pressed = FALSE;
|
|
867
|
|
868 /* clear sampling parameter displays */
|
|
869 textbox_set_text(mainwin_rate_text, " ");
|
|
870 textbox_set_text(mainwin_freq_text, " ");
|
|
871 monostereo_set_num_channels(mainwin_monostereo, 0);
|
|
872
|
|
873 playstatus_set_status(mainwin_playstatus, STATUS_STOP);
|
|
874
|
|
875 /* hide playback time */
|
|
876 widget_hide(WIDGET(mainwin_minus_num));
|
|
877 widget_hide(WIDGET(mainwin_10min_num));
|
|
878 widget_hide(WIDGET(mainwin_min_num));
|
|
879 widget_hide(WIDGET(mainwin_10sec_num));
|
|
880 widget_hide(WIDGET(mainwin_sec_num));
|
|
881
|
|
882 textbox_set_text(mainwin_stime_min, " ");
|
|
883 textbox_set_text(mainwin_stime_sec, " ");
|
|
884
|
|
885 widget_hide(WIDGET(mainwin_position));
|
|
886 widget_hide(WIDGET(mainwin_sposition));
|
|
887
|
|
888 playlistwin_hide_timer();
|
|
889 draw_main_window(TRUE);
|
|
890
|
|
891 vis_clear(active_vis);
|
|
892 }
|
|
893
|
|
894 void
|
|
895 mainwin_disable_seekbar(void)
|
|
896 {
|
|
897 /*
|
|
898 * We dont call draw_main_window() here so this will not
|
|
899 * remove them visually. It will only prevent us from sending
|
|
900 * any seek calls to the input plugin before the input plugin
|
|
901 * calls ->set_info().
|
|
902 */
|
|
903 widget_hide(WIDGET(mainwin_position));
|
|
904 widget_hide(WIDGET(mainwin_sposition));
|
|
905 }
|
|
906
|
|
907 static gboolean
|
|
908 mainwin_mouse_button_release(GtkWidget * widget,
|
|
909 GdkEventButton * event,
|
|
910 gpointer callback_data)
|
|
911 {
|
|
912 gdk_pointer_ungrab(GDK_CURRENT_TIME);
|
|
913
|
|
914 /*
|
|
915 * The gdk_flush() is just for making sure that the pointer really
|
|
916 * gets ungrabbed before calling any button callbacks
|
|
917 *
|
|
918 */
|
|
919
|
|
920 gdk_flush();
|
|
921
|
|
922 if (dock_is_moving(GTK_WINDOW(mainwin))) {
|
|
923 dock_move_release(GTK_WINDOW(mainwin));
|
|
924 }
|
|
925
|
|
926 if (mainwin_menurow->mr_doublesize_selected) {
|
|
927 event->x /= 2;
|
|
928 event->y /= 2;
|
|
929 }
|
|
930
|
|
931 handle_release_cb(mainwin_wlist, widget, event);
|
|
932
|
|
933 draw_main_window(FALSE);
|
|
934
|
|
935 return FALSE;
|
|
936 }
|
|
937
|
|
938 static gboolean
|
|
939 mainwin_motion(GtkWidget * widget,
|
|
940 GdkEventMotion * event,
|
|
941 gpointer callback_data)
|
|
942 {
|
|
943 XEvent ev;
|
|
944 gint i = 0;
|
|
945
|
|
946 XSync(GDK_DISPLAY(), False);
|
|
947
|
|
948 while (XCheckTypedEvent(GDK_DISPLAY(), MotionNotify, &ev)) {
|
|
949 event->x = ev.xmotion.x;
|
|
950 event->y = ev.xmotion.y;
|
|
951 i++;
|
|
952 }
|
|
953
|
|
954 if (dock_is_moving(GTK_WINDOW(mainwin))) {
|
|
955 dock_move_motion(GTK_WINDOW(mainwin), event);
|
|
956 }
|
|
957 else {
|
|
958 handle_motion_cb(mainwin_wlist, widget, event);
|
|
959 draw_main_window(FALSE);
|
|
960 }
|
|
961 gdk_flush();
|
|
962
|
|
963 return FALSE;
|
|
964 }
|
|
965
|
|
966 static gboolean
|
|
967 inside_sensitive_widgets(gint x, gint y)
|
|
968 {
|
|
969 return (widget_contains(WIDGET(mainwin_menubtn), x, y)
|
|
970 || widget_contains(WIDGET(mainwin_minimize), x, y)
|
|
971 || widget_contains(WIDGET(mainwin_shade), x, y)
|
|
972 || widget_contains(WIDGET(mainwin_close), x, y)
|
|
973 || widget_contains(WIDGET(mainwin_rew), x, y)
|
|
974 || widget_contains(WIDGET(mainwin_play), x, y)
|
|
975 || widget_contains(WIDGET(mainwin_pause), x, y)
|
|
976 || widget_contains(WIDGET(mainwin_stop), x, y)
|
|
977 || widget_contains(WIDGET(mainwin_fwd), x, y)
|
|
978 || widget_contains(WIDGET(mainwin_eject), x, y)
|
|
979 || widget_contains(WIDGET(mainwin_shuffle), x, y)
|
|
980 || widget_contains(WIDGET(mainwin_repeat), x, y)
|
|
981 || widget_contains(WIDGET(mainwin_pl), x, y)
|
|
982 || widget_contains(WIDGET(mainwin_eq), x, y)
|
|
983 || widget_contains(WIDGET(mainwin_info), x, y)
|
|
984 || widget_contains(WIDGET(mainwin_menurow), x, y)
|
|
985 || widget_contains(WIDGET(mainwin_volume), x, y)
|
|
986 || widget_contains(WIDGET(mainwin_balance), x, y)
|
|
987 || (widget_contains(WIDGET(mainwin_position), x, y) &&
|
|
988 widget_is_visible(WIDGET(mainwin_position)))
|
|
989 || widget_contains(WIDGET(mainwin_minus_num), x, y)
|
|
990 || widget_contains(WIDGET(mainwin_10min_num), x, y)
|
|
991 || widget_contains(WIDGET(mainwin_min_num), x, y)
|
|
992 || widget_contains(WIDGET(mainwin_10sec_num), x, y)
|
|
993 || widget_contains(WIDGET(mainwin_sec_num), x, y)
|
|
994 || widget_contains(WIDGET(mainwin_vis), x, y)
|
|
995 || widget_contains(WIDGET(mainwin_minimize), x, y)
|
|
996 || widget_contains(WIDGET(mainwin_shade), x, y)
|
|
997 || widget_contains(WIDGET(mainwin_close), x, y)
|
|
998 || widget_contains(WIDGET(mainwin_menubtn), x, y)
|
|
999 || widget_contains(WIDGET(mainwin_sposition), x, y)
|
|
1000 || widget_contains(WIDGET(mainwin_stime_min), x, y)
|
|
1001 || widget_contains(WIDGET(mainwin_stime_sec), x, y)
|
|
1002 || widget_contains(WIDGET(mainwin_srew), x, y)
|
|
1003 || widget_contains(WIDGET(mainwin_splay), x, y)
|
|
1004 || widget_contains(WIDGET(mainwin_spause), x, y)
|
|
1005 || widget_contains(WIDGET(mainwin_sstop), x, y)
|
|
1006 || widget_contains(WIDGET(mainwin_sfwd), x, y)
|
|
1007 || widget_contains(WIDGET(mainwin_seject), x, y)
|
|
1008 || widget_contains(WIDGET(mainwin_svis), x, y)
|
|
1009 || widget_contains(WIDGET(mainwin_about), x, y));
|
|
1010 }
|
|
1011
|
|
1012 void
|
|
1013 mainwin_scrolled(GtkWidget * widget,
|
|
1014 GdkEventScroll * event,
|
|
1015 gpointer callback_data)
|
|
1016 {
|
|
1017 gint d = cfg.mouse_change;
|
|
1018 if (event->direction == GDK_SCROLL_DOWN)
|
|
1019 d *= -1;
|
|
1020 mainwin_set_volume_diff(d);
|
|
1021 }
|
|
1022
|
|
1023
|
|
1024 static gboolean
|
|
1025 mainwin_mouse_button_press(GtkWidget * widget,
|
|
1026 GdkEventButton * event,
|
|
1027 gpointer callback_data)
|
|
1028 {
|
|
1029
|
|
1030 gboolean grab = TRUE;
|
|
1031
|
|
1032 if (event->button == 1 && event->type == GDK_BUTTON_PRESS &&
|
|
1033 !inside_sensitive_widgets(event->x, event->y) && event->y < 14) {
|
|
1034 if (0 && hint_move_resize_available()) {
|
|
1035 hint_move_resize(mainwin, event->x_root, event->y_root, TRUE);
|
|
1036 grab = FALSE;
|
|
1037 }
|
|
1038 else {
|
|
1039 gtk_window_present(GTK_WINDOW(mainwin));
|
|
1040 dock_move_press(dock_window_list, GTK_WINDOW(mainwin), event,
|
|
1041 TRUE);
|
|
1042 }
|
|
1043 }
|
|
1044 else if (event->button == 1 && event->type == GDK_2BUTTON_PRESS &&
|
|
1045 event->y < 14 && !inside_sensitive_widgets(event->x, event->y)) {
|
|
1046 mainwin_set_shade(!cfg.player_shaded);
|
|
1047 if (dock_is_moving(GTK_WINDOW(mainwin)))
|
|
1048 dock_move_release(GTK_WINDOW(mainwin));
|
|
1049 }
|
|
1050 else if (event->button == 1 && event->type == GDK_2BUTTON_PRESS &&
|
|
1051 widget_contains(WIDGET(mainwin_info), event->x, event->y)) {
|
|
1052 playlist_fileinfo_current();
|
|
1053 }
|
|
1054 else {
|
|
1055 handle_press_cb(mainwin_wlist, widget, event);
|
|
1056 draw_main_window(FALSE);
|
|
1057 }
|
|
1058
|
|
1059 if ((event->button == 1) && event->type != GDK_2BUTTON_PRESS &&
|
|
1060 (widget_contains(WIDGET(mainwin_vis), event->x, event->y) ||
|
|
1061 widget_contains(WIDGET(mainwin_svis), event->x, event->y))) {
|
|
1062
|
|
1063 cfg.vis_type++;
|
|
1064
|
|
1065 if (cfg.vis_type > VIS_OFF)
|
|
1066 cfg.vis_type = VIS_ANALYZER;
|
|
1067
|
|
1068 mainwin_vis_set_type(cfg.vis_type);
|
|
1069 }
|
|
1070
|
|
1071 if (event->button == 3) {
|
|
1072 if (widget_contains(WIDGET(mainwin_info), event->x, event->y)) {
|
|
1073 util_item_factory_popup(mainwin_songname_menu,
|
|
1074 event->x_root, event->y_root,
|
|
1075 3, event->time);
|
|
1076 grab = FALSE;
|
|
1077 }
|
|
1078 else if (widget_contains(WIDGET(mainwin_vis), event->x, event->y) ||
|
|
1079 widget_contains(WIDGET(mainwin_svis), event->x, event->y)) {
|
|
1080 util_item_factory_popup(mainwin_vis_menu, event->x_root,
|
|
1081 event->y_root, 3, event->time);
|
|
1082 grab = FALSE;
|
|
1083 }
|
|
1084 else if ( (event->y > 70) && (event->x < 128) )
|
|
1085 {
|
|
1086
|
|
1087 util_item_factory_popup(mainwin_play_menu,
|
|
1088 event->x_root,
|
|
1089 event->y_root, 3, event->time);
|
|
1090 grab = FALSE;
|
|
1091 } else {
|
|
1092 /*
|
|
1093 * Pop up the main menu a few pixels down.
|
|
1094 * This will avoid that anything is selected
|
|
1095 * if one right-clicks to focus the window
|
|
1096 * without raising it.
|
|
1097 *
|
|
1098 ***MD I think the above is stupid, people don't expect this
|
|
1099 *
|
|
1100 */
|
|
1101 util_item_factory_popup(mainwin_general_menu,
|
|
1102 event->x_root,
|
|
1103 event->y_root, 3, event->time);
|
|
1104 grab = FALSE;
|
|
1105 }
|
|
1106 }
|
|
1107 if (event->button == 1) {
|
|
1108 if ((event->x > 35 && event->x < 100 &&
|
|
1109 event->y > 25 && event->y < 40) ||
|
|
1110 widget_contains(WIDGET(mainwin_stime_min), event->x, event->y) ||
|
|
1111 widget_contains(WIDGET(mainwin_stime_sec), event->x, event->y)) {
|
|
1112
|
|
1113 if (cfg.timer_mode == TIMER_ELAPSED)
|
|
1114 set_timer_mode(TIMER_REMAINING);
|
|
1115 else
|
|
1116 set_timer_mode(TIMER_ELAPSED);
|
|
1117 }
|
|
1118
|
|
1119 }
|
|
1120
|
|
1121 if (grab)
|
|
1122 gdk_pointer_grab(mainwin->window, FALSE,
|
|
1123 GDK_BUTTON_MOTION_MASK |
|
|
1124 GDK_BUTTON_RELEASE_MASK,
|
|
1125 GDK_WINDOW(GDK_NONE), NULL, GDK_CURRENT_TIME);
|
|
1126
|
|
1127 return FALSE;
|
|
1128 }
|
|
1129
|
|
1130 static gboolean
|
|
1131 mainwin_focus_in(GtkWidget * window,
|
|
1132 GdkEventFocus * event,
|
|
1133 gpointer data)
|
|
1134 {
|
|
1135 mainwin_menubtn->pb_allow_draw = TRUE;
|
|
1136 mainwin_minimize->pb_allow_draw = TRUE;
|
|
1137 mainwin_shade->pb_allow_draw = TRUE;
|
|
1138 mainwin_close->pb_allow_draw = TRUE;
|
|
1139 draw_main_window(TRUE);
|
|
1140
|
|
1141 return TRUE;
|
|
1142 }
|
|
1143
|
|
1144
|
|
1145 static gboolean
|
|
1146 mainwin_focus_out(GtkWidget * widget,
|
|
1147 GdkEventFocus * event,
|
|
1148 gpointer callback_data)
|
|
1149 {
|
|
1150 mainwin_menubtn->pb_allow_draw = FALSE;
|
|
1151 mainwin_minimize->pb_allow_draw = FALSE;
|
|
1152 mainwin_shade->pb_allow_draw = FALSE;
|
|
1153 mainwin_close->pb_allow_draw = FALSE;
|
|
1154 draw_main_window(TRUE);
|
|
1155
|
|
1156 return TRUE;
|
|
1157 }
|
|
1158
|
|
1159 static gboolean
|
|
1160 mainwin_keypress(GtkWidget * grab_widget,
|
|
1161 GdkEventKey * event,
|
|
1162 gpointer data)
|
|
1163 {
|
|
1164
|
|
1165 switch (event->keyval) {
|
|
1166
|
|
1167 case GDK_Up:
|
|
1168 case GDK_KP_Up:
|
|
1169 mainwin_set_volume_diff(2);
|
|
1170 break;
|
|
1171 case GDK_Down:
|
|
1172 case GDK_KP_Down:
|
|
1173 mainwin_set_volume_diff(-2);
|
|
1174 break;
|
|
1175 case GDK_Left:
|
|
1176 case GDK_KP_Left:
|
|
1177 if (playlist_get_current_length() != -1)
|
|
1178 bmp_playback_seek(CLAMP
|
|
1179 (bmp_playback_get_time() - 5000, 0,
|
|
1180 playlist_get_current_length()) / 1000);
|
|
1181 break;
|
|
1182 case GDK_Right:
|
|
1183 case GDK_KP_Right:
|
|
1184 if (playlist_get_current_length() != -1)
|
|
1185 bmp_playback_seek(CLAMP
|
|
1186 (bmp_playback_get_time() + 5000, 0,
|
|
1187 playlist_get_current_length()) / 1000);
|
|
1188 break;
|
|
1189 case GDK_Escape:
|
|
1190 mainwin_minimize_cb();
|
|
1191 break;
|
|
1192 default:
|
|
1193 return FALSE;
|
|
1194 }
|
|
1195
|
|
1196 return TRUE;
|
|
1197 }
|
|
1198
|
|
1199 static void
|
|
1200 mainwin_jump_to_time_cb(GtkWidget * widget,
|
|
1201 GtkWidget * entry)
|
|
1202 {
|
|
1203 guint min = 0, sec = 0, params, time;
|
|
1204
|
|
1205 params = sscanf(gtk_entry_get_text(GTK_ENTRY(entry)), "%u:%u",
|
|
1206 &min, &sec);
|
|
1207 if (params == 2)
|
|
1208 time = (min * 60) + sec;
|
|
1209 else if (params == 1)
|
|
1210 time = min;
|
|
1211 else
|
|
1212 return;
|
|
1213
|
|
1214 if (playlist_get_current_length() > -1 &&
|
|
1215 time <= (playlist_get_current_length() / 1000)) {
|
|
1216 bmp_playback_seek(time);
|
|
1217 gtk_widget_destroy(mainwin_jtt);
|
|
1218 }
|
|
1219 }
|
|
1220
|
|
1221
|
|
1222 void
|
|
1223 mainwin_jump_to_time(void)
|
|
1224 {
|
|
1225 GtkWidget *vbox, *hbox_new, *hbox_total;
|
|
1226 GtkWidget *time_entry, *label, *bbox, *jump, *cancel;
|
|
1227 guint tindex;
|
|
1228 gchar time_str[10];
|
|
1229
|
|
1230 if (!bmp_playback_get_playing()) {
|
|
1231 /* FIXME: pop an error dialog and/or disable menu option to
|
|
1232 indicate JTT can't be launched when no track is being
|
|
1233 played */
|
|
1234 return;
|
|
1235 }
|
|
1236
|
|
1237 if (mainwin_jtt) {
|
|
1238 gtk_window_present(GTK_WINDOW(mainwin_jtt));
|
|
1239 return;
|
|
1240 }
|
|
1241
|
|
1242 mainwin_jtt = gtk_window_new(GTK_WINDOW_TOPLEVEL);
|
|
1243 gtk_window_set_type_hint(GTK_WINDOW(mainwin_jtt),
|
|
1244 GDK_WINDOW_TYPE_HINT_DIALOG);
|
|
1245
|
|
1246 gtk_window_set_title(GTK_WINDOW(mainwin_jtt), _("Jump to Time"));
|
|
1247 gtk_window_set_position(GTK_WINDOW(mainwin_jtt), GTK_WIN_POS_CENTER);
|
|
1248 gtk_window_set_transient_for(GTK_WINDOW(mainwin_jtt),
|
|
1249 GTK_WINDOW(mainwin));
|
|
1250
|
|
1251 g_signal_connect(mainwin_jtt, "destroy",
|
|
1252 G_CALLBACK(gtk_widget_destroyed), &mainwin_jtt);
|
|
1253 gtk_container_border_width(GTK_CONTAINER(mainwin_jtt), 10);
|
|
1254
|
|
1255 vbox = gtk_vbox_new(FALSE, 5);
|
|
1256 gtk_container_add(GTK_CONTAINER(mainwin_jtt), vbox);
|
|
1257
|
|
1258 hbox_new = gtk_hbox_new(FALSE, 0);
|
|
1259 gtk_box_pack_start(GTK_BOX(vbox), hbox_new, TRUE, TRUE, 5);
|
|
1260
|
|
1261 time_entry = gtk_entry_new();
|
|
1262 gtk_box_pack_start(GTK_BOX(hbox_new), time_entry, FALSE, FALSE, 5);
|
|
1263 g_signal_connect(time_entry, "activate",
|
|
1264 G_CALLBACK(mainwin_jump_to_time_cb), time_entry);
|
|
1265
|
|
1266 gtk_widget_set_size_request(time_entry, 70, -1);
|
|
1267 label = gtk_label_new(_("minutes:seconds"));
|
|
1268 gtk_box_pack_start(GTK_BOX(hbox_new), label, FALSE, FALSE, 5);
|
|
1269
|
|
1270 hbox_total = gtk_hbox_new(FALSE, 0);
|
|
1271 gtk_box_pack_start(GTK_BOX(vbox), hbox_total, TRUE, TRUE, 5);
|
|
1272 gtk_widget_show(hbox_total);
|
|
1273
|
|
1274 /* FIXME: Disable display of current track length. It's not
|
|
1275 updated when track changes */
|
|
1276 #if 0
|
|
1277 label = gtk_label_new(_("Track length:"));
|
|
1278 gtk_box_pack_start(GTK_BOX(hbox_total), label, FALSE, FALSE, 5);
|
|
1279
|
|
1280 len = playlist_get_current_length() / 1000;
|
|
1281 g_snprintf(time_str, sizeof(time_str), "%u:%2.2u", len / 60, len % 60);
|
|
1282 label = gtk_label_new(time_str);
|
|
1283
|
|
1284 gtk_box_pack_start(GTK_BOX(hbox_total), label, FALSE, FALSE, 10);
|
|
1285 #endif
|
|
1286
|
|
1287 bbox = gtk_hbutton_box_new();
|
|
1288 gtk_box_pack_start(GTK_BOX(vbox), bbox, TRUE, TRUE, 0);
|
|
1289 gtk_button_box_set_layout(GTK_BUTTON_BOX(bbox), GTK_BUTTONBOX_END);
|
|
1290 gtk_button_box_set_spacing(GTK_BUTTON_BOX(bbox), 5);
|
|
1291
|
|
1292 cancel = gtk_button_new_from_stock(GTK_STOCK_CANCEL);
|
|
1293 GTK_WIDGET_SET_FLAGS(cancel, GTK_CAN_DEFAULT);
|
|
1294 gtk_container_add(GTK_CONTAINER(bbox), cancel);
|
|
1295 g_signal_connect_swapped(cancel, "clicked",
|
|
1296 G_CALLBACK(gtk_widget_destroy), mainwin_jtt);
|
|
1297
|
|
1298 jump = gtk_button_new_from_stock(GTK_STOCK_JUMP_TO);
|
|
1299 GTK_WIDGET_SET_FLAGS(jump, GTK_CAN_DEFAULT);
|
|
1300 gtk_container_add(GTK_CONTAINER(bbox), jump);
|
|
1301 g_signal_connect(jump, "clicked",
|
|
1302 G_CALLBACK(mainwin_jump_to_time_cb), time_entry);
|
|
1303
|
|
1304 tindex = bmp_playback_get_time() / 1000;
|
|
1305 g_snprintf(time_str, sizeof(time_str), "%u:%2.2u", tindex / 60,
|
|
1306 tindex % 60);
|
|
1307 gtk_entry_set_text(GTK_ENTRY(time_entry), time_str);
|
|
1308
|
|
1309 gtk_entry_select_region(GTK_ENTRY(time_entry), 0, strlen(time_str));
|
|
1310
|
|
1311 gtk_widget_show_all(mainwin_jtt);
|
|
1312
|
|
1313 gtk_widget_grab_focus(time_entry);
|
|
1314 gtk_widget_grab_default(jump);
|
|
1315 }
|
|
1316
|
|
1317 static void
|
|
1318 change_song(guint pos)
|
|
1319 {
|
|
1320 if (bmp_playback_get_playing())
|
|
1321 bmp_playback_stop();
|
|
1322
|
|
1323 playlist_set_position(pos);
|
|
1324 bmp_playback_initiate();
|
|
1325 }
|
|
1326
|
|
1327 static void
|
|
1328 mainwin_jump_to_file_jump(GtkTreeView * treeview)
|
|
1329 {
|
|
1330 GtkTreeModel *model;
|
|
1331 GtkTreeSelection *selection;
|
|
1332 GtkTreeIter iter;
|
|
1333 gchar *pos_str;
|
|
1334 guint pos;
|
|
1335
|
|
1336 model = gtk_tree_view_get_model(treeview);
|
|
1337 selection = gtk_tree_view_get_selection(treeview);
|
|
1338
|
|
1339 if (!gtk_tree_selection_get_selected(selection, NULL, &iter))
|
|
1340 return;
|
|
1341
|
|
1342 gtk_tree_model_get(model, &iter, 0, &pos_str, -1);
|
|
1343 pos = g_ascii_strtoull(pos_str, NULL, 10) - 1;
|
|
1344
|
|
1345 change_song(pos);
|
|
1346
|
|
1347 /* FIXME: should only hide window */
|
|
1348 gtk_widget_destroy(mainwin_jtf);
|
|
1349 mainwin_jtf = NULL;
|
|
1350 }
|
|
1351
|
|
1352 static void
|
|
1353 mainwin_jump_to_file_jump_cb(GtkTreeView * treeview,
|
|
1354 gpointer data)
|
|
1355 {
|
|
1356 mainwin_jump_to_file_jump(treeview);
|
|
1357 }
|
|
1358
|
|
1359 static void
|
|
1360 mainwin_jump_to_file_set_queue_button_label(GtkButton * button,
|
|
1361 guint pos)
|
|
1362 {
|
|
1363 if (playlist_is_position_queued(pos))
|
|
1364 gtk_button_set_label(button, _("Un_queue"));
|
|
1365 else
|
|
1366 gtk_button_set_label(button, _("_Queue"));
|
|
1367 }
|
|
1368
|
|
1369 static void
|
|
1370 mainwin_jump_to_file_queue_cb(GtkButton * button,
|
|
1371 gpointer data)
|
|
1372 {
|
|
1373 GtkTreeView *treeview;
|
|
1374 GtkTreeModel *model;
|
|
1375 GtkTreeSelection *selection;
|
|
1376 GtkTreeIter iter;
|
|
1377 gchar *pos_str;
|
|
1378 guint pos;
|
|
1379
|
|
1380 treeview = GTK_TREE_VIEW(data);
|
|
1381 model = gtk_tree_view_get_model(treeview);
|
|
1382 selection = gtk_tree_view_get_selection(treeview);
|
|
1383
|
|
1384 if (!gtk_tree_selection_get_selected(selection, NULL, &iter))
|
|
1385 return;
|
|
1386
|
|
1387 gtk_tree_model_get(model, &iter, 0, &pos_str, -1);
|
|
1388 pos = g_ascii_strtoull(pos_str, NULL, 10) - 1;
|
|
1389
|
|
1390 playlist_queue_position(pos);
|
|
1391
|
|
1392 mainwin_jump_to_file_set_queue_button_label(button, pos);
|
|
1393 }
|
|
1394
|
|
1395 static void
|
|
1396 mainwin_jump_to_file_selection_changed_cb(GtkTreeSelection *treesel,
|
|
1397 gpointer data)
|
|
1398 {
|
|
1399 GtkTreeView *treeview;
|
|
1400 GtkTreeModel *model;
|
|
1401 GtkTreeSelection *selection;
|
|
1402 GtkTreeIter iter;
|
|
1403 gchar *pos_str;
|
|
1404 guint pos;
|
|
1405
|
|
1406 treeview = gtk_tree_selection_get_tree_view(treesel);
|
|
1407 model = gtk_tree_view_get_model(treeview);
|
|
1408 selection = gtk_tree_view_get_selection(treeview);
|
|
1409
|
|
1410 if (!gtk_tree_selection_get_selected(selection, NULL, &iter))
|
|
1411 return;
|
|
1412
|
|
1413 gtk_tree_model_get(model, &iter, 0, &pos_str, -1);
|
|
1414 pos = g_ascii_strtoull(pos_str, NULL, 10) - 1;
|
|
1415
|
|
1416 mainwin_jump_to_file_set_queue_button_label(GTK_BUTTON(data), pos);
|
|
1417 }
|
|
1418
|
|
1419 static gboolean
|
|
1420 mainwin_jump_to_file_keypress_cb(GtkWidget * object,
|
|
1421 GdkEventKey * event,
|
|
1422 gpointer data)
|
|
1423 {
|
|
1424 switch (event->keyval) {
|
|
1425 case GDK_Escape:
|
|
1426 /* FIXME: show only hide window */
|
|
1427 gtk_widget_destroy(mainwin_jtf);
|
|
1428 mainwin_jtf = NULL;
|
|
1429 return TRUE;
|
|
1430 case GDK_Return:
|
|
1431 mainwin_jump_to_file_jump(GTK_TREE_VIEW(data));
|
|
1432 return TRUE;
|
|
1433 default:
|
|
1434 return FALSE;
|
|
1435 };
|
|
1436 }
|
|
1437
|
|
1438 static gboolean
|
|
1439 mainwin_jump_to_file_match(const gchar * song, gchar ** keys)
|
|
1440 {
|
|
1441 gint i = 0;
|
|
1442 gchar *key;
|
|
1443 gchar *song_lc;
|
|
1444
|
|
1445 song_lc = g_ascii_strdown(song, -1);
|
|
1446
|
|
1447 while (keys[i]) {
|
|
1448 key = g_ascii_strdown(keys[i], -1);
|
|
1449 if (!g_strrstr(song_lc, key)) {
|
|
1450 g_free(key);
|
|
1451 g_free(song_lc);
|
|
1452 return FALSE;
|
|
1453 }
|
|
1454
|
|
1455 g_free(key);
|
|
1456 i++;
|
|
1457 }
|
|
1458
|
|
1459 g_free(song_lc);
|
|
1460
|
|
1461 return TRUE;
|
|
1462 }
|
|
1463
|
|
1464 /* FIXME: Clear the entry when the list gets updated */
|
|
1465 static void
|
|
1466 mainwin_update_jtf(GtkWidget * widget, gpointer user_data)
|
|
1467 {
|
|
1468 /* FIXME: Is not in sync with playlist due to delayed extinfo
|
|
1469 * reading */
|
|
1470 gint row;
|
|
1471 GList *playlist;
|
|
1472 gchar *desc_buf;
|
|
1473 gchar *row_str;
|
|
1474 GtkTreeIter iter;
|
|
1475 GtkTreeSelection *selection;
|
|
1476
|
|
1477 GtkTreeModel *store;
|
|
1478
|
|
1479 if (!mainwin_jtf)
|
|
1480 return;
|
|
1481
|
|
1482 store = gtk_tree_view_get_model(GTK_TREE_VIEW(user_data));
|
|
1483 gtk_list_store_clear(GTK_LIST_STORE(store));
|
|
1484
|
|
1485 row = 1;
|
|
1486 for (playlist = playlist_get(); playlist;
|
|
1487 playlist = g_list_next(playlist)) {
|
|
1488 PlaylistEntry *entry = PLAYLIST_ENTRY(playlist->data);
|
|
1489
|
|
1490 if (entry->title)
|
|
1491 desc_buf = entry->title;
|
|
1492 else if (strchr(entry->filename, '/'))
|
|
1493 desc_buf = strrchr(entry->filename, '/') + 1;
|
|
1494 else
|
|
1495 desc_buf = entry->filename;
|
|
1496
|
|
1497 row_str = g_strdup_printf("%d", row++);
|
|
1498
|
|
1499 gtk_list_store_append(GTK_LIST_STORE(store), &iter);
|
|
1500 gtk_list_store_set(GTK_LIST_STORE(store), &iter,
|
|
1501 0, row_str, 1, desc_buf, -1);
|
|
1502
|
|
1503 g_free(row_str);
|
|
1504 }
|
|
1505
|
|
1506 gtk_tree_model_get_iter_first(GTK_TREE_MODEL(store), &iter);
|
|
1507 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(user_data));
|
|
1508 gtk_tree_selection_select_iter(selection, &iter);
|
|
1509 }
|
|
1510
|
|
1511 static void
|
|
1512 mainwin_jump_to_file_edit_cb(GtkEntry * entry, gpointer user_data)
|
|
1513 {
|
|
1514 GtkTreeView *treeview = GTK_TREE_VIEW(user_data);
|
|
1515 GtkTreeSelection *selection;
|
|
1516 GtkTreeIter iter;
|
|
1517
|
|
1518 GtkListStore *store;
|
|
1519
|
|
1520 gint song_index = 0;
|
|
1521 gchar **words;
|
|
1522 GList *playlist;
|
|
1523
|
|
1524 gboolean match = FALSE;
|
|
1525
|
|
1526 /* Chop the key string into ' '-separated key words */
|
|
1527 words = g_strsplit(gtk_entry_get_text(entry), " ", 0);
|
|
1528
|
|
1529 /* FIXME: Remove the connected signals before clearing
|
|
1530 * (row-selected will still eventually arrive once) */
|
|
1531 store = GTK_LIST_STORE(gtk_tree_view_get_model(treeview));
|
|
1532 gtk_list_store_clear(store);
|
|
1533
|
|
1534 PLAYLIST_LOCK();
|
|
1535
|
|
1536 for (playlist = playlist_get(); playlist;
|
|
1537 playlist = g_list_next(playlist)) {
|
|
1538
|
|
1539 PlaylistEntry *entry = PLAYLIST_ENTRY(playlist->data);
|
|
1540 const gchar *title, *filename;
|
|
1541
|
|
1542 title = entry->title;
|
|
1543 if (!title) {
|
|
1544 filename = entry->filename;
|
|
1545
|
|
1546 if (strchr(filename, '/'))
|
|
1547 title = strrchr(filename, '/') + 1;
|
|
1548 else
|
|
1549 title = filename;
|
|
1550 }
|
|
1551
|
|
1552 /* Compare the key words to the string - if all the words
|
|
1553 match, add to the ListStore */
|
|
1554
|
|
1555 /*
|
|
1556 * FIXME: The search string should be adapted to the
|
|
1557 * current display setting, e.g. if the user has set it to
|
|
1558 * "%p - %t" then build the match string like that too, or
|
|
1559 * even better, search for each of the tags seperatly.
|
|
1560 *
|
|
1561 * In any case the string to match should _never_ contain
|
|
1562 * something the user can't actually see in the playlist.
|
|
1563 */
|
|
1564 if (words[0])
|
|
1565 match = mainwin_jump_to_file_match(title, words);
|
|
1566 else
|
|
1567 match = TRUE;
|
|
1568
|
|
1569 if (match) {
|
|
1570 gchar *song_index_str = g_strdup_printf("%d", song_index + 1);
|
|
1571 gtk_list_store_append(store, &iter);
|
|
1572 gtk_list_store_set(store, &iter, 0, song_index_str, 1, title, -1);
|
|
1573 g_free(song_index_str);
|
|
1574 }
|
|
1575
|
|
1576 song_index++;
|
|
1577 }
|
|
1578
|
|
1579 PLAYLIST_UNLOCK();
|
|
1580
|
|
1581 g_strfreev(words);
|
|
1582
|
|
1583 if (gtk_tree_model_get_iter_first(GTK_TREE_MODEL(store), &iter)) {
|
|
1584 selection = gtk_tree_view_get_selection(treeview);
|
|
1585 gtk_tree_selection_select_iter(selection, &iter);
|
|
1586 }
|
|
1587 }
|
|
1588
|
|
1589 void
|
|
1590 mainwin_jump_to_file(void)
|
|
1591 {
|
|
1592 GtkWidget *scrollwin;
|
|
1593 GtkWidget *vbox, *bbox, *sep;
|
|
1594 GtkWidget *jump, *queue, *cancel;
|
|
1595 GtkWidget *rescan, *edit;
|
|
1596 GtkWidget *search_label, *hbox;
|
|
1597 GList *playlist;
|
|
1598 gchar *desc_buf;
|
|
1599 gchar *row_str;
|
|
1600 gint row;
|
|
1601
|
|
1602 GtkWidget *treeview;
|
|
1603 GtkListStore *jtf_store;
|
|
1604
|
|
1605 GtkTreeIter iter;
|
|
1606 GtkCellRenderer *renderer;
|
|
1607 GtkTreeViewColumn *column;
|
|
1608
|
|
1609 if (mainwin_jtf) {
|
|
1610 gtk_window_present(GTK_WINDOW(mainwin_jtf));
|
|
1611 return;
|
|
1612 }
|
|
1613
|
|
1614 mainwin_jtf = gtk_window_new(GTK_WINDOW_TOPLEVEL);
|
|
1615 gtk_window_set_type_hint(GTK_WINDOW(mainwin_jtf),
|
|
1616 GDK_WINDOW_TYPE_HINT_DIALOG);
|
|
1617
|
|
1618 gtk_window_set_title(GTK_WINDOW(mainwin_jtf), _("Jump to Track"));
|
|
1619
|
|
1620 gtk_window_set_position(GTK_WINDOW(mainwin_jtf), GTK_WIN_POS_CENTER);
|
|
1621 g_signal_connect(mainwin_jtf, "destroy",
|
|
1622 G_CALLBACK(gtk_widget_destroyed), &mainwin_jtf);
|
|
1623
|
|
1624 gtk_container_border_width(GTK_CONTAINER(mainwin_jtf), 10);
|
|
1625 gtk_window_set_default_size(GTK_WINDOW(mainwin_jtf), 550, 350);
|
|
1626
|
|
1627 vbox = gtk_vbox_new(FALSE, 5);
|
|
1628 gtk_container_add(GTK_CONTAINER(mainwin_jtf), vbox);
|
|
1629
|
|
1630 jtf_store = gtk_list_store_new(2, G_TYPE_STRING, G_TYPE_STRING);
|
|
1631 treeview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(jtf_store));
|
|
1632 g_object_unref(jtf_store);
|
|
1633
|
|
1634 gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(treeview), TRUE);
|
|
1635
|
|
1636 column = gtk_tree_view_column_new();
|
|
1637 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview), FALSE);
|
|
1638 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
|
|
1639
|
|
1640 renderer = gtk_cell_renderer_text_new();
|
|
1641 gtk_tree_view_column_pack_start(column, renderer, FALSE);
|
|
1642 gtk_tree_view_column_set_attributes(column, renderer, "text", 0, NULL);
|
|
1643 gtk_tree_view_column_set_spacing(column, 4);
|
|
1644
|
|
1645 renderer = gtk_cell_renderer_text_new();
|
|
1646 gtk_tree_view_column_pack_start(column, renderer, FALSE);
|
|
1647 gtk_tree_view_column_set_attributes(column, renderer, "text", 1, NULL);
|
|
1648 gtk_tree_view_column_set_spacing(column, 4);
|
|
1649 gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column);
|
|
1650
|
|
1651 g_signal_connect(treeview, "row-activated",
|
|
1652 G_CALLBACK(mainwin_jump_to_file_jump), NULL);
|
|
1653
|
|
1654 hbox = gtk_hbox_new(FALSE, 3);
|
|
1655 gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 3);
|
|
1656
|
|
1657 search_label = gtk_label_new(_("Filter: "));
|
|
1658 gtk_label_set_markup_with_mnemonic(GTK_LABEL(search_label), "_Filter:");
|
|
1659 gtk_box_pack_start(GTK_BOX(hbox), search_label, FALSE, FALSE, 0);
|
|
1660
|
|
1661 edit = gtk_entry_new();
|
|
1662 gtk_entry_set_editable(GTK_ENTRY(edit), TRUE);
|
|
1663 gtk_label_set_mnemonic_widget(GTK_LABEL(search_label), edit);
|
|
1664 g_signal_connect(edit, "changed",
|
|
1665 G_CALLBACK(mainwin_jump_to_file_edit_cb), treeview);
|
|
1666
|
|
1667 g_signal_connect(mainwin_jtf, "key_press_event",
|
|
1668 G_CALLBACK(mainwin_jump_to_file_keypress_cb), treeview);
|
|
1669
|
|
1670 gtk_box_pack_start(GTK_BOX(hbox), edit, TRUE, TRUE, 3);
|
|
1671
|
|
1672 scrollwin = gtk_scrolled_window_new(NULL, NULL);
|
|
1673 gtk_container_add(GTK_CONTAINER(scrollwin), treeview);
|
|
1674 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrollwin),
|
|
1675 GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS);
|
|
1676 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrollwin),
|
|
1677 GTK_SHADOW_IN);
|
|
1678 gtk_box_pack_start(GTK_BOX(vbox), scrollwin, TRUE, TRUE, 0);
|
|
1679
|
|
1680 sep = gtk_hseparator_new();
|
|
1681 gtk_box_pack_start(GTK_BOX(vbox), sep, FALSE, FALSE, 0);
|
|
1682
|
|
1683 bbox = gtk_hbutton_box_new();
|
|
1684 gtk_button_box_set_layout(GTK_BUTTON_BOX(bbox), GTK_BUTTONBOX_END);
|
|
1685 gtk_button_box_set_spacing(GTK_BUTTON_BOX(bbox), 5);
|
|
1686 gtk_box_pack_start(GTK_BOX(vbox), bbox, FALSE, FALSE, 0);
|
|
1687
|
|
1688 queue = gtk_button_new_with_mnemonic(_("_Queue"));
|
|
1689 gtk_box_pack_start(GTK_BOX(bbox), queue, FALSE, FALSE, 0);
|
|
1690 GTK_WIDGET_SET_FLAGS(queue, GTK_CAN_DEFAULT);
|
|
1691 g_signal_connect(queue, "clicked",
|
|
1692 G_CALLBACK(mainwin_jump_to_file_queue_cb),
|
|
1693 treeview);
|
|
1694 g_signal_connect(gtk_tree_view_get_selection(GTK_TREE_VIEW(treeview)), "changed",
|
|
1695 G_CALLBACK(mainwin_jump_to_file_selection_changed_cb),
|
|
1696 queue);
|
|
1697
|
|
1698 rescan = gtk_button_new_from_stock(GTK_STOCK_REFRESH);
|
|
1699 gtk_box_pack_start(GTK_BOX(bbox), rescan, FALSE, FALSE, 0);
|
|
1700 g_signal_connect(rescan, "clicked",
|
|
1701 G_CALLBACK(mainwin_update_jtf), treeview);
|
|
1702 GTK_WIDGET_SET_FLAGS(rescan, GTK_CAN_DEFAULT);
|
|
1703 gtk_widget_grab_default(rescan);
|
|
1704
|
|
1705 jump = gtk_button_new_from_stock(GTK_STOCK_JUMP_TO);
|
|
1706 gtk_box_pack_start(GTK_BOX(bbox), jump, FALSE, FALSE, 0);
|
|
1707
|
|
1708 g_signal_connect_swapped(jump, "clicked",
|
|
1709 G_CALLBACK(mainwin_jump_to_file_jump_cb),
|
|
1710 treeview);
|
|
1711
|
|
1712 GTK_WIDGET_SET_FLAGS(jump, GTK_CAN_DEFAULT);
|
|
1713 gtk_widget_grab_default(jump);
|
|
1714
|
|
1715 cancel = gtk_button_new_from_stock(GTK_STOCK_CLOSE);
|
|
1716 gtk_box_pack_start(GTK_BOX(bbox), cancel, FALSE, FALSE, 0);
|
|
1717 g_signal_connect_swapped(cancel, "clicked",
|
|
1718 G_CALLBACK(gtk_widget_destroy),
|
|
1719 mainwin_jtf);
|
|
1720 GTK_WIDGET_SET_FLAGS(cancel, GTK_CAN_DEFAULT);
|
|
1721
|
|
1722 gtk_list_store_clear(jtf_store);
|
|
1723
|
|
1724 row = 1;
|
|
1725
|
|
1726 PLAYLIST_LOCK();
|
|
1727
|
|
1728 for (playlist = playlist_get(); playlist;
|
|
1729 playlist = g_list_next(playlist)) {
|
|
1730
|
|
1731 PlaylistEntry *entry = PLAYLIST_ENTRY(playlist->data);
|
|
1732
|
|
1733 if (entry->title)
|
|
1734 desc_buf = entry->title;
|
|
1735 else if (strchr(entry->filename, '/'))
|
|
1736 desc_buf = strrchr(entry->filename, '/') + 1;
|
|
1737 else
|
|
1738 desc_buf = entry->filename;
|
|
1739
|
|
1740 row_str = g_strdup_printf("%d", row++);
|
|
1741
|
|
1742 gtk_list_store_append(GTK_LIST_STORE(jtf_store), &iter);
|
|
1743 gtk_list_store_set(GTK_LIST_STORE(jtf_store), &iter,
|
|
1744 0, row_str, 1, desc_buf, -1);
|
|
1745
|
|
1746 g_free(row_str);
|
|
1747 }
|
|
1748
|
|
1749 PLAYLIST_UNLOCK();
|
|
1750
|
|
1751 gtk_widget_show_all(mainwin_jtf);
|
|
1752 }
|
|
1753
|
|
1754 static gboolean
|
|
1755 mainwin_configure(GtkWidget * window,
|
|
1756 GdkEventConfigure * event,
|
|
1757 gpointer data)
|
|
1758 {
|
|
1759 if (!GTK_WIDGET_VISIBLE(window))
|
|
1760 return FALSE;
|
|
1761
|
|
1762 if (cfg.show_wm_decorations)
|
|
1763 gdk_window_get_root_origin(window->window,
|
|
1764 &cfg.player_x, &cfg.player_y);
|
|
1765 else
|
|
1766 gdk_window_get_deskrelative_origin(window->window,
|
|
1767 &cfg.player_x, &cfg.player_y);
|
|
1768 return FALSE;
|
|
1769 }
|
|
1770
|
|
1771 void
|
|
1772 mainwin_set_back_pixmap(void)
|
|
1773 {
|
|
1774 gdk_window_set_back_pixmap(mainwin->window, mainwin_bg, 0);
|
|
1775 gdk_window_clear(mainwin->window);
|
|
1776 }
|
|
1777
|
|
1778 void
|
|
1779 mainwin_drag_data_received(GtkWidget * widget,
|
|
1780 GdkDragContext * context,
|
|
1781 gint x,
|
|
1782 gint y,
|
|
1783 GtkSelectionData * selection_data,
|
|
1784 guint info,
|
|
1785 guint time,
|
|
1786 gpointer user_data)
|
|
1787 {
|
|
1788 ConfigDb *db;
|
|
1789 gchar *path, *decoded;
|
|
1790
|
|
1791 if (!selection_data->data) {
|
|
1792 g_warning("DND data string is NULL");
|
|
1793 return;
|
|
1794 }
|
|
1795
|
|
1796 path = (gchar *) selection_data->data;
|
|
1797
|
|
1798 g_message(path);
|
|
1799
|
|
1800 /* FIXME: use a real URL validator/parser */
|
|
1801
|
|
1802 if (str_has_prefix_nocase(path, "fonts:///")) {
|
|
1803 path[strlen(path) - 2] = 0; /* Why the hell a CR&LF? */
|
|
1804 path += 8;
|
|
1805
|
|
1806 /* plain, since we already stripped the first URI part */
|
|
1807 decoded = xmms_urldecode_plain(path);
|
|
1808
|
|
1809 /* Get the old font's size, and add it to the dropped
|
|
1810 * font's name */
|
|
1811 cfg.playlist_font = g_strconcat(decoded + 1,
|
|
1812 strrchr(cfg.playlist_font, ' '),
|
|
1813 NULL);
|
|
1814 playlist_list_set_font(cfg.playlist_font);
|
|
1815 playlistwin_update_list();
|
|
1816
|
|
1817 g_free(decoded);
|
|
1818 return;
|
|
1819 }
|
|
1820
|
|
1821 if (str_has_prefix_nocase(path, "file:///")) {
|
|
1822 path[strlen(path) - 2] = 0; /* Why the hell a CR&LF? */
|
|
1823 path += 7;
|
|
1824 }
|
|
1825 else if (str_has_prefix_nocase(path, "file:")) {
|
|
1826 path += 5;
|
|
1827 }
|
|
1828
|
|
1829 if (file_is_archive(path)) {
|
|
1830 bmp_active_skin_load(path);
|
|
1831 skin_install_skin(path); /* ...and install the skin */
|
|
1832 skin_view_update(user_data);
|
|
1833 /* Change skin name in the config file */
|
|
1834 db = bmp_cfg_db_open();
|
|
1835 bmp_cfg_db_set_string(db, NULL, "skin", path);
|
|
1836 bmp_cfg_db_close(db);
|
|
1837 }
|
|
1838 else {
|
|
1839 if (input_check_file((gchar *) selection_data->data, FALSE)) {
|
|
1840 playlist_clear();
|
|
1841 playlist_add_url((gchar *) selection_data->data);
|
|
1842 bmp_playback_initiate();
|
|
1843 }
|
|
1844 }
|
|
1845 }
|
|
1846
|
|
1847 static void
|
|
1848 dirbrowser_add_dir(const gchar * dir)
|
|
1849 {
|
|
1850 g_free(cfg.filesel_path);
|
|
1851 cfg.filesel_path = g_strdup(dir);
|
|
1852 playlist_add_dir(dir);
|
|
1853 }
|
|
1854
|
|
1855 static void
|
|
1856 dirbrowser_on_response(GtkFileChooserDialog * dialog,
|
|
1857 gint result,
|
|
1858 gpointer data)
|
|
1859 {
|
|
1860
|
|
1861 gchar *pathname;
|
|
1862
|
|
1863 // gtk_widget_hide(GTK_WIDGET(dialog));
|
|
1864
|
|
1865 switch (result) {
|
|
1866 case GTK_RESPONSE_ACCEPT:
|
|
1867 pathname = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog));
|
|
1868 dirbrowser_add_dir(pathname);
|
|
1869 g_free(pathname);
|
|
1870 break;
|
|
1871
|
|
1872 case GTK_RESPONSE_CLOSE:
|
|
1873 break;
|
|
1874 }
|
|
1875
|
|
1876 }
|
|
1877
|
|
1878
|
|
1879 static GtkWidget *
|
|
1880 dirbrowser_new(const gchar * path)
|
|
1881 {
|
|
1882 GtkWidget *dialog;
|
|
1883
|
|
1884 dialog = gtk_file_chooser_dialog_new(_("Add Folders"),
|
|
1885 GTK_WINDOW(mainwin),
|
|
1886 GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER,
|
|
1887 GTK_STOCK_ADD, GTK_RESPONSE_OK,
|
|
1888 GTK_STOCK_CLOSE, GTK_RESPONSE_CLOSE,
|
|
1889 NULL);
|
|
1890 gtk_window_set_position(GTK_WINDOW(dialog), GTK_WIN_POS_CENTER);
|
|
1891
|
|
1892 gtk_file_chooser_set_current_folder(GTK_FILE_CHOOSER(dialog),
|
|
1893 path);
|
|
1894 g_signal_connect(dialog, "response",
|
|
1895 G_CALLBACK(dirbrowser_on_response),
|
|
1896 NULL);
|
|
1897
|
|
1898 return dialog;
|
|
1899 }
|
|
1900
|
|
1901 void
|
|
1902 mainwin_run_dirbrowser(void)
|
|
1903 {
|
|
1904 static GtkWidget *browser = NULL;
|
|
1905
|
|
1906 if (!browser) {
|
|
1907 browser = dirbrowser_new(cfg.filesel_path);
|
|
1908
|
|
1909 g_signal_connect(browser, "destroy",
|
|
1910 G_CALLBACK(gtk_widget_destroyed),
|
|
1911 &browser);
|
|
1912
|
|
1913 gtk_widget_show(GTK_WIDGET(browser));
|
|
1914 }
|
|
1915
|
|
1916 gtk_window_present(GTK_WINDOW(browser));
|
|
1917 }
|
|
1918
|
|
1919 static void
|
|
1920 on_add_url_add_clicked(GtkWidget * widget,
|
|
1921 GtkWidget * entry)
|
|
1922 {
|
|
1923 const gchar *text = gtk_entry_get_text(GTK_ENTRY(entry));
|
|
1924 if (text && *text)
|
|
1925 playlist_add_url(text);
|
|
1926 }
|
|
1927
|
|
1928 void
|
|
1929 mainwin_show_add_url_window(void)
|
|
1930 {
|
|
1931 static GtkWidget *url_window = NULL;
|
|
1932
|
|
1933 if (!url_window) {
|
|
1934 url_window =
|
|
1935 util_add_url_dialog_new(_("Add Internet Address"),
|
|
1936 G_CALLBACK(on_add_url_add_clicked));
|
|
1937 gtk_window_set_transient_for(GTK_WINDOW(url_window),
|
|
1938 GTK_WINDOW(mainwin));
|
|
1939 g_signal_connect(url_window, "destroy",
|
|
1940 G_CALLBACK(gtk_widget_destroyed),
|
|
1941 &url_window);
|
|
1942 }
|
|
1943
|
|
1944 gtk_window_present(GTK_WINDOW(url_window));
|
|
1945 }
|
|
1946
|
|
1947 static void
|
|
1948 check_set(GtkItemFactory * factory,
|
|
1949 const gchar * path,
|
|
1950 gboolean active)
|
|
1951 {
|
|
1952 GtkWidget *item = gtk_item_factory_get_widget(factory, path);
|
|
1953 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(item), active);
|
|
1954 }
|
|
1955
|
|
1956 void
|
|
1957 mainwin_eject_pushed(void)
|
|
1958 {
|
|
1959 util_run_filebrowser(PLAY_BUTTON);
|
|
1960 }
|
|
1961
|
|
1962 void
|
|
1963 mainwin_play_pushed(void)
|
|
1964 {
|
|
1965 if (bmp_playback_get_paused()) {
|
|
1966 bmp_playback_pause();
|
|
1967 return;
|
|
1968 }
|
|
1969
|
|
1970 if (playlist_get_length())
|
|
1971 bmp_playback_initiate();
|
|
1972 else
|
|
1973 mainwin_eject_pushed();
|
|
1974 }
|
|
1975
|
|
1976 void
|
|
1977 mainwin_stop_pushed(void)
|
|
1978 {
|
|
1979 mainwin_clear_song_info();
|
|
1980 bmp_playback_stop();
|
|
1981 }
|
|
1982
|
|
1983 void
|
|
1984 mainwin_shuffle_pushed(gboolean toggled)
|
|
1985 {
|
|
1986 check_set(mainwin_play_menu, "/Shuffle", toggled);
|
|
1987 }
|
|
1988
|
|
1989 void
|
|
1990 mainwin_repeat_pushed(gboolean toggled)
|
|
1991 {
|
|
1992 check_set(mainwin_play_menu, "/Repeat", toggled);
|
|
1993 }
|
|
1994
|
|
1995 void
|
|
1996 mainwin_pl_pushed(gboolean toggled)
|
|
1997 {
|
|
1998 if (toggled)
|
|
1999 playlistwin_show();
|
|
2000 else
|
|
2001 playlistwin_hide();
|
|
2002 }
|
|
2003
|
|
2004 gint
|
|
2005 mainwin_spos_frame_cb(gint pos)
|
|
2006 {
|
|
2007 if (mainwin_sposition) {
|
|
2008 if (pos < 6)
|
|
2009 mainwin_sposition->hs_knob_nx = mainwin_sposition->hs_knob_px =
|
|
2010 17;
|
|
2011 else if (pos < 9)
|
|
2012 mainwin_sposition->hs_knob_nx = mainwin_sposition->hs_knob_px =
|
|
2013 20;
|
|
2014 else
|
|
2015 mainwin_sposition->hs_knob_nx = mainwin_sposition->hs_knob_px =
|
|
2016 23;
|
|
2017 }
|
|
2018 return 1;
|
|
2019 }
|
|
2020
|
|
2021 void
|
|
2022 mainwin_spos_motion_cb(gint pos)
|
|
2023 {
|
|
2024 gint time;
|
|
2025 gchar *time_msg;
|
|
2026
|
|
2027 pos--;
|
|
2028
|
|
2029 time = ((playlist_get_current_length() / 1000) * pos) / 12;
|
|
2030
|
|
2031 if (cfg.timer_mode == TIMER_REMAINING) {
|
|
2032 time = (playlist_get_current_length() / 1000) - time;
|
|
2033 time_msg = g_strdup_printf("-%2.2d", time / 60);
|
|
2034 textbox_set_text(mainwin_stime_min, time_msg);
|
|
2035 g_free(time_msg);
|
|
2036 }
|
|
2037 else {
|
|
2038 time_msg = g_strdup_printf(" %2.2d", time / 60);
|
|
2039 textbox_set_text(mainwin_stime_min, time_msg);
|
|
2040 g_free(time_msg);
|
|
2041 }
|
|
2042
|
|
2043 time_msg = g_strdup_printf("%2.2d", time % 60);
|
|
2044 textbox_set_text(mainwin_stime_sec, time_msg);
|
|
2045 g_free(time_msg);
|
|
2046 }
|
|
2047
|
|
2048 void
|
|
2049 mainwin_spos_release_cb(gint pos)
|
|
2050 {
|
|
2051 bmp_playback_seek(((playlist_get_current_length() / 1000) *
|
|
2052 (pos - 1)) / 12);
|
|
2053 }
|
|
2054
|
|
2055 void
|
|
2056 mainwin_position_motion_cb(gint pos)
|
|
2057 {
|
|
2058 gint length, time;
|
|
2059 gchar *seek_msg;
|
|
2060
|
|
2061 length = playlist_get_current_length() / 1000;
|
|
2062 time = (length * pos) / 219;
|
|
2063 seek_msg = g_strdup_printf(_("SEEK TO: %d:%-2.2d/%d:%-2.2d (%d%%)"),
|
|
2064 time / 60, time % 60,
|
|
2065 length / 60, length % 60,
|
|
2066 (length != 0) ? (time * 100) / length : 0);
|
|
2067 mainwin_lock_info_text(seek_msg);
|
|
2068 g_free(seek_msg);
|
|
2069 }
|
|
2070
|
|
2071 void
|
|
2072 mainwin_position_release_cb(gint pos)
|
|
2073 {
|
|
2074 gint length, time;
|
|
2075
|
|
2076 length = playlist_get_current_length() / 1000;
|
|
2077 time = (length * pos) / 219;
|
|
2078 bmp_playback_seek(time);
|
|
2079 mainwin_release_info_text();
|
|
2080 }
|
|
2081
|
|
2082 gint
|
|
2083 mainwin_volume_frame_cb(gint pos)
|
|
2084 {
|
|
2085 return (gint) rint((pos / 52.0) * 28);
|
|
2086 }
|
|
2087
|
|
2088 void
|
|
2089 mainwin_adjust_volume_motion(gint v)
|
|
2090 {
|
|
2091 gchar *volume_msg;
|
|
2092
|
|
2093 setting_volume = TRUE;
|
|
2094
|
|
2095 volume_msg = g_strdup_printf(_("VOLUME: %d%%"), v);
|
|
2096 mainwin_lock_info_text(volume_msg);
|
|
2097 g_free(volume_msg);
|
|
2098
|
|
2099 if (balance < 0)
|
|
2100 input_set_volume(v, (v * (100 - abs(balance))) / 100);
|
|
2101 else if (balance > 0)
|
|
2102 input_set_volume((v * (100 - abs(balance))) / 100, v);
|
|
2103 else
|
|
2104 input_set_volume(v, v);
|
|
2105 }
|
|
2106
|
|
2107 void
|
|
2108 mainwin_adjust_volume_release(void)
|
|
2109 {
|
|
2110 mainwin_release_info_text();
|
|
2111 setting_volume = FALSE;
|
|
2112 read_volume(VOLUME_ADJUSTED);
|
|
2113 }
|
|
2114
|
|
2115 void
|
|
2116 mainwin_adjust_balance_motion(gint b)
|
|
2117 {
|
|
2118 gchar *balance_msg;
|
|
2119 gint v, pvl, pvr;
|
|
2120
|
|
2121 setting_volume = TRUE;
|
|
2122 balance = b;
|
|
2123 input_get_volume(&pvl, &pvr);
|
|
2124 v = MAX(pvl, pvr);
|
|
2125 if (b < 0) {
|
|
2126 balance_msg = g_strdup_printf(_("BALANCE: %d%% LEFT"), -b);
|
|
2127 input_set_volume(v, (gint) rint(((100 + b) / 100.0) * v));
|
|
2128 }
|
|
2129 else if (b == 0) {
|
|
2130 balance_msg = g_strdup_printf(_("BALANCE: CENTER"));
|
|
2131 input_set_volume(v, v);
|
|
2132 }
|
|
2133 else { /* b > 0 */
|
|
2134 balance_msg = g_strdup_printf(_("BALANCE: %d%% RIGHT"), b);
|
|
2135 input_set_volume((gint) rint(((100 - b) / 100.0) * v), v);
|
|
2136 }
|
|
2137 mainwin_lock_info_text(balance_msg);
|
|
2138 g_free(balance_msg);
|
|
2139 }
|
|
2140
|
|
2141 void
|
|
2142 mainwin_adjust_balance_release(void)
|
|
2143 {
|
|
2144 mainwin_release_info_text();
|
|
2145 setting_volume = FALSE;
|
|
2146 read_volume(VOLUME_ADJUSTED);
|
|
2147 }
|
|
2148
|
|
2149 void
|
|
2150 mainwin_set_volume_slider(gint percent)
|
|
2151 {
|
|
2152 hslider_set_position(mainwin_volume, (gint) rint((percent * 51) / 100.0));
|
|
2153 }
|
|
2154
|
|
2155 void
|
|
2156 mainwin_set_balance_slider(gint percent)
|
|
2157 {
|
|
2158 hslider_set_position(mainwin_balance,
|
|
2159 (gint) rint(((percent * 12) / 100.0) + 12));
|
|
2160 }
|
|
2161
|
|
2162 void
|
|
2163 mainwin_volume_motion_cb(gint pos)
|
|
2164 {
|
|
2165 gint vol = (pos * 100) / 51;
|
|
2166 mainwin_adjust_volume_motion(vol);
|
|
2167 equalizerwin_set_volume_slider(vol);
|
|
2168 }
|
|
2169
|
|
2170 void
|
|
2171 mainwin_volume_release_cb(gint pos)
|
|
2172 {
|
|
2173 mainwin_adjust_volume_release();
|
|
2174 }
|
|
2175
|
|
2176 gint
|
|
2177 mainwin_balance_frame_cb(gint pos)
|
|
2178 {
|
|
2179 return ((abs(pos - 12) * 28) / 13);
|
|
2180 }
|
|
2181
|
|
2182 void
|
|
2183 mainwin_balance_motion_cb(gint pos)
|
|
2184 {
|
|
2185 gint bal = ((pos - 12) * 100) / 12;
|
|
2186 mainwin_adjust_balance_motion(bal);
|
|
2187 equalizerwin_set_balance_slider(bal);
|
|
2188 }
|
|
2189
|
|
2190 void
|
|
2191 mainwin_balance_release_cb(gint pos)
|
|
2192 {
|
|
2193 mainwin_adjust_volume_release();
|
|
2194 }
|
|
2195
|
|
2196 void
|
|
2197 mainwin_set_volume_diff(gint diff)
|
|
2198 {
|
|
2199 gint vl, vr, vol;
|
|
2200
|
|
2201 input_get_volume(&vl, &vr);
|
|
2202 vol = MAX(vl, vr);
|
|
2203 vol = CLAMP(vol + diff, 0, 100);
|
|
2204
|
|
2205 mainwin_adjust_volume_motion(vol);
|
|
2206 setting_volume = FALSE;
|
|
2207 mainwin_set_volume_slider(vol);
|
|
2208 equalizerwin_set_volume_slider(vol);
|
|
2209 read_volume(VOLUME_SET);
|
|
2210 }
|
|
2211
|
|
2212 void
|
|
2213 mainwin_set_balance_diff(gint diff)
|
|
2214 {
|
|
2215 gint b;
|
|
2216 b = CLAMP(balance + diff, -100, 100);
|
|
2217 mainwin_adjust_balance_motion(b);
|
|
2218 setting_volume = FALSE;
|
|
2219 mainwin_set_balance_slider(b);
|
|
2220 equalizerwin_set_balance_slider(b);
|
|
2221 read_volume(VOLUME_SET);
|
|
2222 }
|
|
2223
|
|
2224 void
|
|
2225 mainwin_show(gboolean show)
|
|
2226 {
|
|
2227 if (show)
|
|
2228 mainwin_real_show();
|
|
2229 else
|
|
2230 mainwin_real_hide();
|
|
2231 }
|
|
2232
|
|
2233 void
|
|
2234 mainwin_real_show(void)
|
|
2235 {
|
|
2236 cfg.player_visible = TRUE;
|
|
2237
|
|
2238 if (cfg.player_shaded)
|
|
2239 vis_clear_data(active_vis);
|
|
2240
|
|
2241 mainwin_vis_set_active_vis(MAINWIN_VIS_ACTIVE_MAINWIN);
|
|
2242 mainwin_set_shape_mask();
|
|
2243
|
|
2244 if (cfg.show_wm_decorations) {
|
|
2245 if (!pposition_broken && cfg.player_x != -1
|
|
2246 && cfg.save_window_position)
|
|
2247 gtk_window_move(GTK_WINDOW(mainwin), cfg.player_x, cfg.player_y);
|
|
2248
|
|
2249 gtk_widget_show(mainwin);
|
|
2250
|
|
2251 if (pposition_broken && cfg.player_x != -1
|
|
2252 && cfg.save_window_position)
|
|
2253 gtk_window_move(GTK_WINDOW(mainwin), cfg.player_x, cfg.player_y);
|
|
2254
|
|
2255 return;
|
|
2256 }
|
|
2257
|
|
2258 gtk_widget_show_all(mainwin);
|
|
2259
|
|
2260 if (!nullmask)
|
|
2261 return;
|
|
2262
|
|
2263 g_object_unref(nullmask);
|
|
2264 nullmask = NULL;
|
|
2265
|
|
2266 gdk_window_set_hints(mainwin->window, 0, 0,
|
|
2267 PLAYER_WIDTH, PLAYER_HEIGHT,
|
|
2268 PLAYER_WIDTH, PLAYER_HEIGHT,
|
|
2269 GDK_HINT_MIN_SIZE | GDK_HINT_MAX_SIZE);
|
|
2270 gtk_window_resize(GTK_WINDOW(mainwin), PLAYER_WIDTH, PLAYER_HEIGHT);
|
|
2271
|
|
2272 if (cfg.player_x != -1 && cfg.player_y != -1)
|
|
2273 gtk_window_move(GTK_WINDOW(mainwin), cfg.player_x, cfg.player_y);
|
|
2274
|
|
2275 draw_main_window(TRUE);
|
|
2276
|
|
2277 gtk_window_present(GTK_WINDOW(mainwin));
|
|
2278 }
|
|
2279
|
|
2280 void
|
|
2281 mainwin_real_hide(void)
|
|
2282 {
|
|
2283 GdkGC *gc;
|
|
2284 GdkColor pattern;
|
|
2285
|
|
2286 if (cfg.player_shaded) {
|
|
2287 svis_clear_data(mainwin_svis);
|
|
2288 vis_clear_data(playlistwin_vis);
|
|
2289 }
|
|
2290
|
|
2291 if (!cfg.show_wm_decorations) {
|
|
2292 nullmask = gdk_pixmap_new(mainwin->window, 20, 20, 1);
|
|
2293 gc = gdk_gc_new(nullmask);
|
|
2294 pattern.pixel = 0;
|
|
2295 gdk_gc_set_foreground(gc, &pattern);
|
|
2296 gdk_draw_rectangle(nullmask, gc, TRUE, 0, 0, 20, 20);
|
|
2297 gdk_gc_destroy(gc);
|
|
2298 gtk_widget_shape_combine_mask(mainwin, nullmask, 0, 0);
|
|
2299
|
|
2300 gdk_window_set_hints(mainwin->window, 0, 0, 0, 0, 0, 0,
|
|
2301 GDK_HINT_MIN_SIZE | GDK_HINT_MAX_SIZE);
|
|
2302 gdk_window_resize(mainwin->window, 0, 0);
|
|
2303 }
|
|
2304
|
|
2305 gtk_widget_hide(mainwin);
|
|
2306
|
|
2307 mainwin_vis_set_active_vis(MAINWIN_VIS_ACTIVE_PLAYLISTWIN);
|
|
2308 cfg.player_visible = FALSE;
|
|
2309 }
|
|
2310
|
|
2311 static void
|
|
2312 mainwin_songname_menu_callback(gpointer data,
|
|
2313 guint action,
|
|
2314 GtkWidget * item)
|
|
2315 {
|
|
2316 GtkCheckMenuItem *check;
|
|
2317
|
|
2318 switch (action) {
|
|
2319 case MAINWIN_SONGNAME_FILEINFO:
|
|
2320 playlist_fileinfo_current();
|
|
2321 break;
|
|
2322 case MAINWIN_SONGNAME_JTF:
|
|
2323 mainwin_jump_to_file();
|
|
2324 break;
|
|
2325 case MAINWIN_SONGNAME_JTT:
|
|
2326 mainwin_jump_to_time();
|
|
2327 break;
|
|
2328 case MAINWIN_SONGNAME_SCROLL:
|
|
2329 check = GTK_CHECK_MENU_ITEM(item);
|
|
2330 mainwin_set_title_scroll(gtk_check_menu_item_get_active(check));
|
|
2331 break;
|
|
2332 }
|
|
2333 }
|
|
2334
|
|
2335 static void
|
|
2336 mainwin_play_menu_callback(gpointer data,
|
|
2337 guint action,
|
|
2338 GtkWidget * item)
|
|
2339 {
|
|
2340 GtkCheckMenuItem *check;
|
|
2341
|
|
2342 switch (action) {
|
|
2343 case MAINWIN_OPT_SHUFFLE:
|
|
2344 check = GTK_CHECK_MENU_ITEM(item);
|
|
2345 cfg.shuffle = gtk_check_menu_item_get_active(check);
|
|
2346 playlist_set_shuffle(cfg.shuffle);
|
|
2347 tbutton_set_toggled(mainwin_shuffle, cfg.shuffle);
|
|
2348 break;
|
|
2349 case MAINWIN_OPT_REPEAT:
|
|
2350 check = GTK_CHECK_MENU_ITEM(item);
|
|
2351 cfg.repeat = gtk_check_menu_item_get_active(check);
|
|
2352 tbutton_set_toggled(mainwin_repeat, cfg.repeat);
|
|
2353 break;
|
|
2354 case MAINWIN_OPT_NPA:
|
|
2355 check = GTK_CHECK_MENU_ITEM(item);
|
|
2356 cfg.no_playlist_advance = gtk_check_menu_item_get_active(check);
|
|
2357 break;
|
|
2358 }
|
|
2359 }
|
|
2360
|
|
2361
|
|
2362 static void
|
|
2363 mainwin_view_menu_callback(gpointer data,
|
|
2364 guint action,
|
|
2365 GtkWidget * item)
|
|
2366 {
|
|
2367 switch (action) {
|
|
2368 case MAINWIN_OPT_TELAPSED:
|
|
2369 set_timer_mode_menu_cb(TIMER_ELAPSED);
|
|
2370 break;
|
|
2371 case MAINWIN_OPT_TREMAINING:
|
|
2372 set_timer_mode_menu_cb(TIMER_REMAINING);
|
|
2373 break;
|
|
2374 case MAINWIN_OPT_ALWAYS:
|
|
2375 mainwin_menurow->mr_always_selected = GTK_CHECK_MENU_ITEM(item)->active;
|
|
2376 cfg.always_on_top = mainwin_menurow->mr_always_selected;
|
|
2377 widget_draw(WIDGET(mainwin_menurow));
|
|
2378 hint_set_always(cfg.always_on_top);
|
|
2379 break;
|
|
2380 case MAINWIN_OPT_STICKY:
|
|
2381 cfg.sticky = GTK_CHECK_MENU_ITEM(item)->active;
|
|
2382 hint_set_sticky(cfg.sticky);
|
|
2383 break;
|
|
2384 case MAINWIN_OPT_WS:
|
|
2385 mainwin_set_shade_menu_cb(GTK_CHECK_MENU_ITEM(item)->active);
|
|
2386 break;
|
|
2387 case MAINWIN_OPT_PWS:
|
|
2388 playlistwin_set_shade(GTK_CHECK_MENU_ITEM(item)->active);
|
|
2389 break;
|
|
2390 case MAINWIN_OPT_EQWS:
|
|
2391 equalizerwin_set_shade_menu_cb(GTK_CHECK_MENU_ITEM(item)->active);
|
|
2392 break;
|
|
2393 }
|
|
2394 }
|
|
2395
|
|
2396 void
|
|
2397 mainwin_vis_menu_callback(gpointer data,
|
|
2398 guint action,
|
|
2399 GtkWidget * item)
|
|
2400 {
|
|
2401 switch (action) {
|
|
2402 case MAINWIN_VIS_ANALYZER:
|
|
2403 case MAINWIN_VIS_SCOPE:
|
|
2404 case MAINWIN_VIS_OFF:
|
|
2405 mainwin_vis_set_type_menu_cb(action - MAINWIN_VIS_ANALYZER);
|
|
2406 break;
|
|
2407 case MAINWIN_VIS_ANALYZER_NORMAL:
|
|
2408 case MAINWIN_VIS_ANALYZER_FIRE:
|
|
2409 case MAINWIN_VIS_ANALYZER_VLINES:
|
|
2410 mainwin_vis_set_analyzer_mode(action - MAINWIN_VIS_ANALYZER_NORMAL);
|
|
2411 break;
|
|
2412 case MAINWIN_VIS_ANALYZER_LINES:
|
|
2413 case MAINWIN_VIS_ANALYZER_BARS:
|
|
2414 mainwin_vis_set_analyzer_type(action - MAINWIN_VIS_ANALYZER_LINES);
|
|
2415 break;
|
|
2416 case MAINWIN_VIS_ANALYZER_PEAKS:
|
|
2417 cfg.analyzer_peaks = GTK_CHECK_MENU_ITEM(item)->active;
|
|
2418 break;
|
|
2419 case MAINWIN_VIS_SCOPE_DOT:
|
|
2420 case MAINWIN_VIS_SCOPE_LINE:
|
|
2421 case MAINWIN_VIS_SCOPE_SOLID:
|
|
2422 cfg.scope_mode = action - MAINWIN_VIS_SCOPE_DOT;
|
|
2423 break;
|
|
2424 case MAINWIN_VIS_VU_NORMAL:
|
|
2425 case MAINWIN_VIS_VU_SMOOTH:
|
|
2426 cfg.vu_mode = action - MAINWIN_VIS_VU_NORMAL;
|
|
2427 break;
|
|
2428 case MAINWIN_VIS_REFRESH_FULL:
|
|
2429 case MAINWIN_VIS_REFRESH_HALF:
|
|
2430 case MAINWIN_VIS_REFRESH_QUARTER:
|
|
2431 case MAINWIN_VIS_REFRESH_EIGHTH:
|
|
2432 mainwin_vis_set_refresh(action - MAINWIN_VIS_REFRESH_FULL);
|
|
2433 break;
|
|
2434 case MAINWIN_VIS_AFALLOFF_SLOWEST:
|
|
2435 case MAINWIN_VIS_AFALLOFF_SLOW:
|
|
2436 case MAINWIN_VIS_AFALLOFF_MEDIUM:
|
|
2437 case MAINWIN_VIS_AFALLOFF_FAST:
|
|
2438 case MAINWIN_VIS_AFALLOFF_FASTEST:
|
|
2439 mainwin_vis_set_afalloff(action - MAINWIN_VIS_AFALLOFF_SLOWEST);
|
|
2440 break;
|
|
2441 case MAINWIN_VIS_PFALLOFF_SLOWEST:
|
|
2442 case MAINWIN_VIS_PFALLOFF_SLOW:
|
|
2443 case MAINWIN_VIS_PFALLOFF_MEDIUM:
|
|
2444 case MAINWIN_VIS_PFALLOFF_FAST:
|
|
2445 case MAINWIN_VIS_PFALLOFF_FASTEST:
|
|
2446 mainwin_vis_set_pfalloff(action - MAINWIN_VIS_PFALLOFF_SLOWEST);
|
|
2447 break;
|
|
2448 }
|
|
2449 }
|
|
2450
|
|
2451 void
|
|
2452 mainwin_general_menu_callback(gpointer data,
|
|
2453 guint action,
|
|
2454 GtkWidget * item)
|
|
2455 {
|
|
2456 switch (action) {
|
|
2457 case MAINWIN_GENERAL_PREFS:
|
|
2458 show_prefs_window();
|
|
2459 break;
|
|
2460 case MAINWIN_GENERAL_ABOUT:
|
|
2461 show_about_window();
|
|
2462 break;
|
|
2463 case MAINWIN_GENERAL_PLAYFILE:
|
|
2464 util_run_filebrowser(NO_PLAY_BUTTON);
|
|
2465 break;
|
|
2466 case MAINWIN_GENERAL_PLAYDIRECTORY:
|
|
2467 mainwin_run_dirbrowser();
|
|
2468 break;
|
|
2469 case MAINWIN_GENERAL_PLAYCD:
|
|
2470 play_medium();
|
|
2471 break;
|
|
2472 case MAINWIN_GENERAL_ADDCD:
|
|
2473 add_medium();
|
|
2474 break;
|
|
2475 case MAINWIN_GENERAL_PLAYLOCATION:
|
|
2476 mainwin_show_add_url_window();
|
|
2477 break;
|
|
2478 case MAINWIN_GENERAL_FILEINFO:
|
|
2479 playlist_fileinfo_current();
|
|
2480 break;
|
|
2481 case MAINWIN_GENERAL_FOCUSPLWIN:
|
|
2482 gtk_window_present(GTK_WINDOW(playlistwin));
|
|
2483 break;
|
|
2484 case MAINWIN_GENERAL_SHOWPLWIN:
|
|
2485 if (GTK_CHECK_MENU_ITEM(item)->active)
|
|
2486 playlistwin_show();
|
|
2487 else
|
|
2488 playlistwin_hide();
|
|
2489 break;
|
|
2490 case MAINWIN_GENERAL_SHOWEQWIN:
|
|
2491 if (GTK_CHECK_MENU_ITEM(item)->active)
|
|
2492 equalizerwin_real_show();
|
|
2493 else
|
|
2494 equalizerwin_real_hide();
|
|
2495 break;
|
|
2496 case MAINWIN_GENERAL_PREV:
|
|
2497 playlist_prev();
|
|
2498 break;
|
|
2499 case MAINWIN_GENERAL_PLAY:
|
|
2500 mainwin_play_pushed();
|
|
2501 break;
|
|
2502 case MAINWIN_GENERAL_PAUSE:
|
|
2503 bmp_playback_pause();
|
|
2504 break;
|
|
2505 case MAINWIN_GENERAL_STOP:
|
|
2506 mainwin_stop_pushed();
|
|
2507 break;
|
|
2508 case MAINWIN_GENERAL_NEXT:
|
|
2509 playlist_next();
|
|
2510 break;
|
|
2511 case MAINWIN_GENERAL_BACK5SEC:
|
|
2512 if (bmp_playback_get_playing()
|
|
2513 && playlist_get_current_length() != -1)
|
|
2514 bmp_playback_seek_relative(-5);
|
|
2515 break;
|
|
2516 case MAINWIN_GENERAL_FWD5SEC:
|
|
2517 if (bmp_playback_get_playing()
|
|
2518 && playlist_get_current_length() != -1)
|
|
2519 bmp_playback_seek_relative(5);
|
|
2520 break;
|
|
2521 case MAINWIN_GENERAL_START:
|
|
2522 playlist_set_position(0);
|
|
2523 break;
|
|
2524 case MAINWIN_GENERAL_JTT:
|
|
2525 mainwin_jump_to_time();
|
|
2526 break;
|
|
2527 case MAINWIN_GENERAL_JTF:
|
|
2528 mainwin_jump_to_file();
|
|
2529 break;
|
|
2530 case MAINWIN_GENERAL_EXIT:
|
|
2531 mainwin_quit_cb();
|
|
2532 break;
|
|
2533 }
|
|
2534 }
|
|
2535
|
|
2536 static void
|
|
2537 mainwin_mr_change(MenuRowItem i)
|
|
2538 {
|
|
2539 switch (i) {
|
|
2540 case MENUROW_NONE:
|
|
2541 mainwin_set_info_text();
|
|
2542 break;
|
|
2543 case MENUROW_OPTIONS:
|
|
2544 mainwin_lock_info_text(_("OPTIONS MENU"));
|
|
2545 break;
|
|
2546 case MENUROW_ALWAYS:
|
|
2547 if (!hint_always_on_top_available()) {
|
|
2548 if (mainwin_menurow->mr_always_selected)
|
|
2549 mainwin_lock_info_text(_("DISABLE ALWAYS ON TOP (N/A)"));
|
|
2550 else
|
|
2551 mainwin_lock_info_text(_("ENABLE ALWAYS ON TOP (N/A)"));
|
|
2552 }
|
|
2553 else if (mainwin_menurow->mr_doublesize_selected)
|
|
2554 mainwin_lock_info_text(_("DISABLE ALWAYS ON TOP"));
|
|
2555 else
|
|
2556 mainwin_lock_info_text(_("ENABLE ALWAYS ON TOP"));
|
|
2557 break;
|
|
2558 case MENUROW_FILEINFOBOX:
|
|
2559 mainwin_lock_info_text(_("FILE INFO BOX"));
|
|
2560 break;
|
|
2561 case MENUROW_DOUBLESIZE:
|
|
2562 mainwin_lock_info_text(_("** DOUBLESIZE HAS BEEN REMOVED **"));
|
|
2563 break;
|
|
2564 case MENUROW_VISUALIZATION:
|
|
2565 mainwin_lock_info_text(_("VISUALIZATION MENU"));
|
|
2566 break;
|
|
2567 }
|
|
2568 }
|
|
2569
|
|
2570 static void
|
|
2571 mainwin_mr_release(MenuRowItem i)
|
|
2572 {
|
|
2573 GdkModifierType modmask;
|
|
2574 GtkWidget *widget;
|
|
2575 gint x, y;
|
|
2576
|
|
2577 switch (i) {
|
|
2578 case MENUROW_OPTIONS:
|
|
2579 gdk_window_get_pointer(NULL, &x, &y, &modmask);
|
|
2580 util_item_factory_popup(mainwin_view_menu, x, y, 1,
|
|
2581 GDK_CURRENT_TIME);
|
|
2582 break;
|
|
2583 case MENUROW_ALWAYS:
|
|
2584 widget =
|
|
2585 gtk_item_factory_get_widget(mainwin_view_menu,
|
|
2586 "/Always On Top");
|
|
2587 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(widget),
|
|
2588 mainwin_menurow->mr_always_selected);
|
|
2589 break;
|
|
2590 case MENUROW_FILEINFOBOX:
|
|
2591 playlist_fileinfo_current();
|
|
2592 break;
|
|
2593 case MENUROW_DOUBLESIZE:
|
|
2594 /* double size removed, do nothing */
|
|
2595 break;
|
|
2596 case MENUROW_VISUALIZATION:
|
|
2597 gdk_window_get_pointer(NULL, &x, &y, &modmask);
|
|
2598 util_item_factory_popup(mainwin_vis_menu, x, y, 1, GDK_CURRENT_TIME);
|
|
2599 break;
|
|
2600 case MENUROW_NONE:
|
|
2601 break;
|
|
2602 }
|
|
2603 mainwin_release_info_text();
|
|
2604 }
|
|
2605
|
|
2606 static void
|
|
2607 run_no_audiocd_dialog(void)
|
|
2608 {
|
|
2609 const gchar *markup =
|
|
2610 N_("<b><big>No playable CD found.</big></b>\n\n"
|
|
2611 "No CD inserted, or inserted CD is not an audio CD.\n");
|
|
2612
|
|
2613 GtkWidget *dialog =
|
|
2614 gtk_message_dialog_new_with_markup(GTK_WINDOW(mainwin),
|
|
2615 GTK_DIALOG_DESTROY_WITH_PARENT,
|
|
2616 GTK_MESSAGE_ERROR,
|
|
2617 GTK_BUTTONS_OK,
|
|
2618 _(markup));
|
|
2619 gtk_dialog_run(GTK_DIALOG(dialog));
|
|
2620 gtk_widget_destroy(dialog);
|
|
2621 }
|
|
2622
|
|
2623 static void
|
|
2624 run_no_output_device_dialog(void)
|
|
2625 {
|
|
2626 const gchar *markup =
|
|
2627 N_("<b><big>Couldn't open audio.</big></b>\n\n"
|
|
2628 "Please check that:\n"
|
|
2629 "1. You have the correct output plugin selected.\n"
|
|
2630 "2. No other programs is blocking the soundcard.\n"
|
|
2631 "3. Your soundcard is configured properly.\n");
|
|
2632
|
|
2633 GtkWidget *dialog =
|
|
2634 gtk_message_dialog_new_with_markup(GTK_WINDOW(mainwin),
|
|
2635 GTK_DIALOG_DESTROY_WITH_PARENT,
|
|
2636 GTK_MESSAGE_ERROR,
|
|
2637 GTK_BUTTONS_OK,
|
|
2638 _(markup));
|
|
2639 gtk_dialog_run(GTK_DIALOG(dialog));
|
|
2640 gtk_widget_destroy(dialog);
|
|
2641 }
|
|
2642
|
|
2643
|
|
2644 void
|
|
2645 add_medium(void)
|
|
2646 {
|
|
2647 GList *list, *node;
|
|
2648 gchar *filename;
|
|
2649 gchar *path;
|
|
2650 ConfigDb *db;
|
|
2651
|
|
2652 db = bmp_cfg_db_open();
|
|
2653
|
|
2654 bmp_cfg_db_get_string(db, "CDDA", "directory", &path);
|
|
2655 bmp_cfg_db_close(db);
|
|
2656
|
|
2657 if (!(list = input_scan_dir(path))) {
|
|
2658 run_no_audiocd_dialog();
|
|
2659 return;
|
|
2660 }
|
|
2661
|
|
2662 for (node = list; node; node = g_list_next(node)) {
|
|
2663 filename = g_build_filename(path, node->data, NULL);
|
|
2664 playlist_add(filename);
|
|
2665 g_free(filename);
|
|
2666 g_free(node->data);
|
|
2667 }
|
|
2668
|
|
2669 g_free(path);
|
|
2670 g_list_free(list);
|
|
2671
|
|
2672 }
|
|
2673
|
|
2674 void
|
|
2675 play_medium(void)
|
|
2676 {
|
|
2677 GList *list, *node;
|
|
2678 gchar *filename;
|
|
2679 gchar *path;
|
|
2680 ConfigDb *db;
|
|
2681
|
|
2682 db = bmp_cfg_db_open();
|
|
2683 bmp_cfg_db_get_string(db, "CDDA", "directory", &path);
|
|
2684 bmp_cfg_db_close(db);
|
|
2685
|
|
2686 if (!(list = input_scan_dir(path))) {
|
|
2687 run_no_audiocd_dialog();
|
|
2688 return;
|
|
2689 }
|
|
2690
|
|
2691 playlist_clear();
|
|
2692
|
|
2693 for (node = list; node; node = g_list_next(node)) {
|
|
2694 filename = g_build_filename(path, node->data, NULL);
|
|
2695 playlist_add(filename);
|
|
2696 g_free(filename);
|
|
2697 g_free(node->data);
|
|
2698 }
|
|
2699
|
|
2700 g_free(path);
|
|
2701 g_list_free(list);
|
|
2702
|
|
2703 playlist_set_position(0);
|
|
2704 bmp_playback_initiate();
|
|
2705 }
|
|
2706
|
|
2707 void
|
|
2708 read_volume(gint when)
|
|
2709 {
|
|
2710 static gint pvl = 0, pvr = 0;
|
|
2711 static gint times = VOLSET_DISP_TIMES;
|
|
2712 static gboolean changing = FALSE;
|
|
2713
|
|
2714 gint vl, vr, b, v;
|
|
2715
|
|
2716 input_get_volume(&vl, &vr);
|
|
2717
|
|
2718 switch (when) {
|
|
2719 case VOLSET_STARTUP:
|
|
2720 vl = CLAMP(vl, 0, 100);
|
|
2721 vr = CLAMP(vr, 0, 100);
|
|
2722 pvl = vl;
|
|
2723 pvr = vr;
|
|
2724 v = MAX(vl, vr);
|
|
2725 if (vl > vr)
|
|
2726 b = (gint) rint(((gdouble) vr / vl) * 100) - 100;
|
|
2727 else if (vl < vr)
|
|
2728 b = 100 - (gint) rint(((gdouble) vl / vr) * 100);
|
|
2729 else
|
|
2730 b = 0;
|
|
2731
|
|
2732 balance = b;
|
|
2733 mainwin_set_volume_slider(v);
|
|
2734 equalizerwin_set_volume_slider(v);
|
|
2735 mainwin_set_balance_slider(b);
|
|
2736 equalizerwin_set_balance_slider(b);
|
|
2737 return;
|
|
2738
|
|
2739 case VOLSET_UPDATE:
|
|
2740 if (vl == -1 || vr == -1)
|
|
2741 return;
|
|
2742
|
|
2743 if (setting_volume) {
|
|
2744 pvl = vl;
|
|
2745 pvr = vr;
|
|
2746 return;
|
|
2747 }
|
|
2748
|
|
2749 if (pvr == vr && pvl == vl && changing) {
|
|
2750 if (times < VOLSET_DISP_TIMES)
|
|
2751 times++;
|
|
2752 else {
|
|
2753 mainwin_release_info_text();
|
|
2754 changing = FALSE;
|
|
2755 }
|
|
2756 }
|
|
2757 else if (pvr != vr || pvl != vl) {
|
|
2758 gchar *tmp;
|
|
2759
|
|
2760 v = MAX(vl, vr);
|
|
2761 if (vl > vr)
|
|
2762 b = (gint) rint(((gdouble) vr / vl) * 100) - 100;
|
|
2763 else if (vl < vr)
|
|
2764 b = 100 - (gint) rint(((gdouble) vl / vr) * 100);
|
|
2765 else
|
|
2766 b = 0;
|
|
2767
|
|
2768 if (MAX(vl, vr) != MAX(pvl, pvr))
|
|
2769 tmp = g_strdup_printf(_("VOLUME: %d%%"), v);
|
|
2770 else {
|
|
2771 if (vl > vr) {
|
|
2772 tmp = g_strdup_printf(_("BALANCE: %d%% LEFT"), -b);
|
|
2773 }
|
|
2774 else if (vr == vl)
|
|
2775 tmp = g_strdup_printf(_("BALANCE: CENTER"));
|
|
2776 else { /* (vl < vr) */
|
|
2777 tmp = g_strdup_printf(_("BALANCE: %d%% RIGHT"), b);
|
|
2778 }
|
|
2779 }
|
|
2780 mainwin_lock_info_text(tmp);
|
|
2781 g_free(tmp);
|
|
2782
|
|
2783 pvr = vr;
|
|
2784 pvl = vl;
|
|
2785 times = 0;
|
|
2786 changing = TRUE;
|
|
2787 mainwin_set_volume_slider(v);
|
|
2788 equalizerwin_set_volume_slider(v);
|
|
2789
|
|
2790 /* Don't change the balance slider if the volume has been
|
|
2791 * set to zero. The balance can be anything, and our best
|
|
2792 * guess is what is was before. */
|
|
2793 if (v > 0) {
|
|
2794 balance = b;
|
|
2795 mainwin_set_balance_slider(b);
|
|
2796 equalizerwin_set_balance_slider(b);
|
|
2797 }
|
|
2798 }
|
|
2799 break;
|
|
2800
|
|
2801 case VOLUME_ADJUSTED:
|
|
2802 pvl = vl;
|
|
2803 pvr = vr;
|
|
2804 break;
|
|
2805
|
|
2806 case VOLUME_SET:
|
|
2807 times = 0;
|
|
2808 changing = TRUE;
|
|
2809 pvl = vl;
|
|
2810 pvr = vr;
|
|
2811 break;
|
|
2812 }
|
|
2813 }
|
|
2814
|
|
2815
|
|
2816 /* TODO: HAL! */
|
|
2817 gboolean
|
|
2818 can_play_cd(void)
|
|
2819 {
|
|
2820 GList *ilist;
|
|
2821
|
|
2822 for (ilist = get_input_list(); ilist; ilist = g_list_next(ilist)) {
|
|
2823 InputPlugin *ip = INPUT_PLUGIN(ilist->data);
|
|
2824
|
|
2825 if (!g_ascii_strcasecmp(g_basename(ip->filename),
|
|
2826 PLUGIN_FILENAME("cdaudio"))) {
|
|
2827 return TRUE;
|
|
2828 }
|
|
2829 }
|
|
2830
|
|
2831 return FALSE;
|
|
2832 }
|
|
2833
|
|
2834
|
|
2835 static void
|
|
2836 set_timer_mode(TimerMode mode)
|
|
2837 {
|
|
2838 if (mode == TIMER_ELAPSED)
|
|
2839 check_set(mainwin_view_menu, "/Time Elapsed", TRUE);
|
|
2840 else
|
|
2841 check_set(mainwin_view_menu, "/Time Remaining", TRUE);
|
|
2842 }
|
|
2843
|
|
2844 static void
|
|
2845 set_timer_mode_menu_cb(TimerMode mode)
|
|
2846 {
|
|
2847 cfg.timer_mode = mode;
|
|
2848 }
|
|
2849
|
|
2850
|
|
2851 void
|
|
2852 mainwin_setup_menus(void)
|
|
2853 {
|
|
2854 set_timer_mode(cfg.timer_mode);
|
|
2855
|
|
2856 /* View menu */
|
|
2857
|
|
2858 check_set(mainwin_view_menu, "/Always On Top", cfg.always_on_top);
|
|
2859 check_set(mainwin_view_menu, "/Put on All Workspaces", cfg.sticky);
|
|
2860 check_set(mainwin_view_menu, "/Roll up Player", cfg.player_shaded);
|
|
2861 check_set(mainwin_view_menu, "/Roll up Playlist Editor", cfg.playlist_shaded);
|
|
2862 check_set(mainwin_view_menu, "/Roll up Equalizer", cfg.equalizer_shaded);
|
|
2863
|
|
2864 /* Songname menu */
|
|
2865
|
|
2866 check_set(mainwin_songname_menu, "/Autoscroll Songname", cfg.autoscroll);
|
|
2867
|
|
2868 /* Playback menu */
|
|
2869
|
|
2870 check_set(mainwin_play_menu, "/Repeat", cfg.repeat);
|
|
2871 check_set(mainwin_play_menu, "/Shuffle", cfg.shuffle);
|
|
2872
|
|
2873 /* Visualization menu */
|
|
2874
|
|
2875 check_set(mainwin_vis_menu,
|
|
2876 mainwin_vis_menu_entries[MAINWIN_VIS_MENU_VIS_MODE +
|
|
2877 cfg.vis_type].path, TRUE);
|
|
2878 check_set(mainwin_vis_menu,
|
|
2879 mainwin_vis_menu_entries[MAINWIN_VIS_MENU_ANALYZER_MODE +
|
|
2880 cfg.analyzer_mode].path, TRUE);
|
|
2881 check_set(mainwin_vis_menu,
|
|
2882 mainwin_vis_menu_entries[MAINWIN_VIS_MENU_ANALYZER_TYPE +
|
|
2883 cfg.analyzer_type].path, TRUE);
|
|
2884 check_set(mainwin_vis_menu,
|
|
2885 mainwin_vis_menu_entries[MAINWIN_VIS_MENU_ANALYZER_PEAKS].
|
|
2886 path, cfg.analyzer_peaks);
|
|
2887 check_set(mainwin_vis_menu,
|
|
2888 mainwin_vis_menu_entries[MAINWIN_VIS_MENU_SCOPE_MODE +
|
|
2889 cfg.scope_mode].path, TRUE);
|
|
2890 check_set(mainwin_vis_menu,
|
|
2891 mainwin_vis_menu_entries[MAINWIN_VIS_MENU_WSHADEVU_MODE +
|
|
2892 cfg.vu_mode].path, TRUE);
|
|
2893 check_set(mainwin_vis_menu,
|
|
2894 mainwin_vis_menu_entries[MAINWIN_VIS_MENU_REFRESH_RATE +
|
|
2895 cfg.vis_refresh].path, TRUE);
|
|
2896 check_set(mainwin_vis_menu,
|
|
2897 mainwin_vis_menu_entries[MAINWIN_VIS_MENU_AFALLOFF +
|
|
2898 cfg.analyzer_falloff].path, TRUE);
|
|
2899 check_set(mainwin_vis_menu,
|
|
2900 mainwin_vis_menu_entries[MAINWIN_VIS_MENU_PFALLOFF +
|
|
2901 cfg.peaks_falloff].path, TRUE);
|
|
2902 }
|
|
2903
|
|
2904 static void
|
|
2905 mainwin_create_widgets(void)
|
|
2906 {
|
|
2907 mainwin_menubtn =
|
|
2908 create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 6, 3, 9, 9,
|
|
2909 0, 0, 0, 9, mainwin_menubtn_cb, SKIN_TITLEBAR);
|
|
2910 mainwin_menubtn->pb_allow_draw = FALSE;
|
|
2911 mainwin_minimize =
|
|
2912 create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 244, 3, 9,
|
|
2913 9, 9, 0, 9, 9, mainwin_minimize_cb, SKIN_TITLEBAR);
|
|
2914 mainwin_minimize->pb_allow_draw = FALSE;
|
|
2915 mainwin_shade =
|
|
2916 create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 254, 3, 9,
|
|
2917 9, 0, cfg.player_shaded ? 27 : 18, 9,
|
|
2918 cfg.player_shaded ? 27 : 18, mainwin_shade_toggle,
|
|
2919 SKIN_TITLEBAR);
|
|
2920 mainwin_shade->pb_allow_draw = FALSE;
|
|
2921 mainwin_close =
|
|
2922 create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 264, 3, 9,
|
|
2923 9, 18, 0, 18, 9, mainwin_quit_cb, SKIN_TITLEBAR);
|
|
2924 mainwin_close->pb_allow_draw = FALSE;
|
|
2925
|
|
2926 mainwin_rew =
|
|
2927 create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 16, 88, 23,
|
|
2928 18, 0, 0, 0, 18, playlist_prev, SKIN_CBUTTONS);
|
|
2929 mainwin_play =
|
|
2930 create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 39, 88, 23,
|
|
2931 18, 23, 0, 23, 18, mainwin_play_pushed, SKIN_CBUTTONS);
|
|
2932 mainwin_pause =
|
|
2933 create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 62, 88, 23,
|
|
2934 18, 46, 0, 46, 18, bmp_playback_pause, SKIN_CBUTTONS);
|
|
2935 mainwin_stop =
|
|
2936 create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 85, 88, 23,
|
|
2937 18, 69, 0, 69, 18, mainwin_stop_pushed, SKIN_CBUTTONS);
|
|
2938 mainwin_fwd =
|
|
2939 create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 108, 88, 22,
|
|
2940 18, 92, 0, 92, 18, playlist_next, SKIN_CBUTTONS);
|
|
2941 mainwin_eject =
|
|
2942 create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 136, 89, 22,
|
|
2943 16, 114, 0, 114, 16, mainwin_eject_pushed,
|
|
2944 SKIN_CBUTTONS);
|
|
2945
|
|
2946 mainwin_srew =
|
|
2947 create_sbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 169, 4, 8,
|
|
2948 7, playlist_prev);
|
|
2949 mainwin_splay =
|
|
2950 create_sbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 177, 4, 10,
|
|
2951 7, mainwin_play_pushed);
|
|
2952 mainwin_spause =
|
|
2953 create_sbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 187, 4, 10,
|
|
2954 7, bmp_playback_pause);
|
|
2955 mainwin_sstop =
|
|
2956 create_sbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 197, 4, 9,
|
|
2957 7, mainwin_stop_pushed);
|
|
2958 mainwin_sfwd =
|
|
2959 create_sbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 206, 4, 8,
|
|
2960 7, playlist_next);
|
|
2961 mainwin_seject =
|
|
2962 create_sbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 216, 4, 9,
|
|
2963 7, mainwin_eject_pushed);
|
|
2964
|
|
2965 mainwin_shuffle =
|
|
2966 create_tbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 164, 89, 46,
|
|
2967 15, 28, 0, 28, 15, 28, 30, 28, 45,
|
|
2968 mainwin_shuffle_pushed, SKIN_SHUFREP);
|
|
2969
|
|
2970 mainwin_repeat =
|
|
2971 create_tbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 210, 89, 28,
|
|
2972 15, 0, 0, 0, 15, 0, 30, 0, 45,
|
|
2973 mainwin_repeat_pushed, SKIN_SHUFREP);
|
|
2974
|
|
2975 mainwin_eq =
|
|
2976 create_tbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 219, 58, 23,
|
|
2977 12, 0, 61, 46, 61, 0, 73, 46, 73, equalizerwin_show,
|
|
2978 SKIN_SHUFREP);
|
|
2979 tbutton_set_toggled(mainwin_eq, cfg.equalizer_visible);
|
|
2980 mainwin_pl =
|
|
2981 create_tbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 242, 58, 23,
|
|
2982 12, 23, 61, 69, 61, 23, 73, 69, 73,
|
|
2983 mainwin_pl_pushed, SKIN_SHUFREP);
|
|
2984 tbutton_set_toggled(mainwin_pl, cfg.playlist_visible);
|
|
2985
|
|
2986 mainwin_info =
|
|
2987 create_textbox(&mainwin_wlist, mainwin_bg, mainwin_gc, 112, 27,
|
|
2988 153, 1, SKIN_TEXT);
|
|
2989 textbox_set_scroll(mainwin_info, cfg.autoscroll);
|
|
2990 textbox_set_xfont(mainwin_info, TRUE, cfg.mainwin_font);
|
|
2991 mainwin_rate_text =
|
|
2992 create_textbox(&mainwin_wlist, mainwin_bg, mainwin_gc, 111, 43, 15,
|
|
2993 0, SKIN_TEXT);
|
|
2994 mainwin_freq_text =
|
|
2995 create_textbox(&mainwin_wlist, mainwin_bg, mainwin_gc, 156, 43, 10,
|
|
2996 0, SKIN_TEXT);
|
|
2997
|
|
2998 mainwin_menurow =
|
|
2999 create_menurow(&mainwin_wlist, mainwin_bg, mainwin_gc, 10, 22, 304,
|
|
3000 0, 304, 44, mainwin_mr_change, mainwin_mr_release,
|
|
3001 SKIN_TITLEBAR);
|
|
3002 mainwin_menurow->mr_doublesize_selected = FALSE;
|
|
3003 mainwin_menurow->mr_always_selected = cfg.always_on_top;
|
|
3004
|
|
3005 mainwin_volume =
|
|
3006 create_hslider(&mainwin_wlist, mainwin_bg, mainwin_gc, 107, 57, 68,
|
|
3007 13, 15, 422, 0, 422, 14, 11, 15, 0, 0, 51,
|
|
3008 mainwin_volume_frame_cb, mainwin_volume_motion_cb,
|
|
3009 mainwin_volume_release_cb, SKIN_VOLUME);
|
|
3010 mainwin_balance =
|
|
3011 create_hslider(&mainwin_wlist, mainwin_bg, mainwin_gc, 177, 57, 38,
|
|
3012 13, 15, 422, 0, 422, 14, 11, 15, 9, 0, 24,
|
|
3013 mainwin_balance_frame_cb, mainwin_balance_motion_cb,
|
|
3014 mainwin_balance_release_cb, SKIN_BALANCE);
|
|
3015
|
|
3016 mainwin_monostereo =
|
|
3017 create_monostereo(&mainwin_wlist, mainwin_bg, mainwin_gc, 212, 41,
|
|
3018 SKIN_MONOSTEREO);
|
|
3019
|
|
3020 mainwin_playstatus =
|
|
3021 create_playstatus(&mainwin_wlist, mainwin_bg, mainwin_gc, 24, 28);
|
|
3022
|
|
3023 mainwin_minus_num =
|
|
3024 create_number(&mainwin_wlist, mainwin_bg, mainwin_gc, 36, 26,
|
|
3025 SKIN_NUMBERS);
|
|
3026 widget_hide(WIDGET(mainwin_minus_num));
|
|
3027 mainwin_10min_num =
|
|
3028 create_number(&mainwin_wlist, mainwin_bg, mainwin_gc, 48, 26,
|
|
3029 SKIN_NUMBERS);
|
|
3030 widget_hide(WIDGET(mainwin_10min_num));
|
|
3031
|
|
3032 mainwin_min_num =
|
|
3033 create_number(&mainwin_wlist, mainwin_bg, mainwin_gc, 60, 26,
|
|
3034 SKIN_NUMBERS);
|
|
3035 widget_hide(WIDGET(mainwin_min_num));
|
|
3036
|
|
3037 mainwin_10sec_num =
|
|
3038 create_number(&mainwin_wlist, mainwin_bg, mainwin_gc, 78, 26,
|
|
3039 SKIN_NUMBERS);
|
|
3040 widget_hide(WIDGET(mainwin_10sec_num));
|
|
3041
|
|
3042 mainwin_sec_num =
|
|
3043 create_number(&mainwin_wlist, mainwin_bg, mainwin_gc, 90, 26,
|
|
3044 SKIN_NUMBERS);
|
|
3045 widget_hide(WIDGET(mainwin_sec_num));
|
|
3046
|
|
3047 mainwin_about =
|
|
3048 create_sbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 247, 83, 20,
|
|
3049 25, show_about_window);
|
|
3050
|
|
3051 mainwin_vis =
|
|
3052 create_vis(&mainwin_wlist, mainwin_bg, mainwin->window, mainwin_gc,
|
|
3053 24, 43, 76);
|
|
3054 mainwin_svis = create_svis(&mainwin_wlist, mainwin_bg, mainwin_gc, 79, 5);
|
|
3055 active_vis = mainwin_vis;
|
|
3056
|
|
3057 mainwin_position =
|
|
3058 create_hslider(&mainwin_wlist, mainwin_bg, mainwin_gc, 16, 72, 248,
|
|
3059 10, 248, 0, 278, 0, 29, 10, 10, 0, 0, 219, NULL,
|
|
3060 mainwin_position_motion_cb,
|
|
3061 mainwin_position_release_cb, SKIN_POSBAR);
|
|
3062 widget_hide(WIDGET(mainwin_position));
|
|
3063
|
|
3064 mainwin_sposition =
|
|
3065 create_hslider(&mainwin_wlist, mainwin_bg, mainwin_gc, 226, 4, 17,
|
|
3066 7, 17, 36, 17, 36, 3, 7, 36, 0, 1, 13,
|
|
3067 mainwin_spos_frame_cb, mainwin_spos_motion_cb,
|
|
3068 mainwin_spos_release_cb, SKIN_TITLEBAR);
|
|
3069 widget_hide(WIDGET(mainwin_sposition));
|
|
3070
|
|
3071 mainwin_stime_min =
|
|
3072 create_textbox(&mainwin_wlist, mainwin_bg, mainwin_gc, 130, 4, 15,
|
|
3073 FALSE, SKIN_TEXT);
|
|
3074 mainwin_stime_sec =
|
|
3075 create_textbox(&mainwin_wlist, mainwin_bg, mainwin_gc, 147, 4, 10,
|
|
3076 FALSE, SKIN_TEXT);
|
|
3077
|
|
3078 if (!cfg.player_shaded) {
|
|
3079 widget_hide(WIDGET(mainwin_svis));
|
|
3080 widget_hide(WIDGET(mainwin_srew));
|
|
3081 widget_hide(WIDGET(mainwin_splay));
|
|
3082 widget_hide(WIDGET(mainwin_spause));
|
|
3083 widget_hide(WIDGET(mainwin_sstop));
|
|
3084 widget_hide(WIDGET(mainwin_sfwd));
|
|
3085 widget_hide(WIDGET(mainwin_seject));
|
|
3086 widget_hide(WIDGET(mainwin_stime_min));
|
|
3087 widget_hide(WIDGET(mainwin_stime_sec));
|
|
3088 }
|
|
3089
|
|
3090 }
|
|
3091
|
|
3092 static void
|
|
3093 mainwin_create_window(void)
|
|
3094 {
|
|
3095 gint width, height;
|
|
3096
|
|
3097 mainwin = gtk_window_new(GTK_WINDOW_TOPLEVEL);
|
|
3098 gtk_window_set_title(GTK_WINDOW(mainwin), _("Audacious"));
|
|
3099 gtk_window_set_role(GTK_WINDOW(mainwin), "player");
|
|
3100 gtk_window_set_resizable(GTK_WINDOW(mainwin), FALSE);
|
|
3101
|
|
3102 width = MAINWIN_WIDTH;
|
|
3103 height = cfg.player_shaded ? MAINWIN_SHADED_HEIGHT : MAINWIN_HEIGHT;
|
|
3104
|
|
3105 gtk_widget_set_size_request(mainwin, width, height);
|
|
3106 gtk_widget_set_app_paintable(mainwin, TRUE);
|
|
3107
|
|
3108 dock_window_list = dock_window_set_decorated(dock_window_list,
|
|
3109 GTK_WINDOW(mainwin),
|
|
3110 cfg.show_wm_decorations);
|
|
3111
|
|
3112 if (cfg.player_x != -1 && cfg.save_window_position)
|
|
3113 gtk_window_move(GTK_WINDOW(mainwin), cfg.player_x, cfg.player_y);
|
|
3114
|
|
3115 gtk_widget_add_events(mainwin,
|
|
3116 GDK_FOCUS_CHANGE_MASK | GDK_BUTTON_MOTION_MASK |
|
|
3117 GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK |
|
|
3118 GDK_SCROLL_MASK | GDK_KEY_PRESS_MASK |
|
|
3119 GDK_VISIBILITY_NOTIFY_MASK);
|
|
3120 gtk_widget_realize(mainwin);
|
|
3121
|
|
3122 util_set_cursor(mainwin);
|
|
3123
|
|
3124 g_signal_connect(mainwin, "destroy", G_CALLBACK(mainwin_destroy), NULL);
|
|
3125 g_signal_connect(mainwin, "button_press_event",
|
|
3126 G_CALLBACK(mainwin_mouse_button_press), NULL);
|
|
3127 g_signal_connect(mainwin, "scroll_event",
|
|
3128 G_CALLBACK(mainwin_scrolled), NULL);
|
|
3129 g_signal_connect(mainwin, "button_release_event",
|
|
3130 G_CALLBACK(mainwin_mouse_button_release), NULL);
|
|
3131 g_signal_connect(mainwin, "motion_notify_event",
|
|
3132 G_CALLBACK(mainwin_motion), NULL);
|
|
3133 g_signal_connect_after(mainwin, "focus_in_event",
|
|
3134 G_CALLBACK(mainwin_focus_in), NULL);
|
|
3135 g_signal_connect_after(mainwin, "focus_out_event",
|
|
3136 G_CALLBACK(mainwin_focus_out), NULL);
|
|
3137 g_signal_connect(mainwin, "configure_event",
|
|
3138 G_CALLBACK(mainwin_configure), NULL);
|
|
3139 g_signal_connect(mainwin, "style_set",
|
|
3140 G_CALLBACK(mainwin_set_back_pixmap), NULL);
|
|
3141
|
|
3142 bmp_drag_dest_set(mainwin);
|
|
3143
|
|
3144 g_signal_connect(mainwin, "key_press_event",
|
|
3145 G_CALLBACK(mainwin_keypress), NULL);
|
|
3146 }
|
|
3147
|
|
3148 static void
|
|
3149 mainwin_create_menus(void)
|
|
3150 {
|
|
3151 mainwin_general_menu = create_menu(mainwin_general_menu_entries,
|
|
3152 mainwin_general_menu_entries_num,
|
|
3153 mainwin_accel);
|
|
3154
|
|
3155 mainwin_play_menu = create_menu(mainwin_playback_menu_entries,
|
|
3156 mainwin_playback_menu_entries_num,
|
|
3157 mainwin_accel);
|
|
3158
|
|
3159 mainwin_view_menu = create_menu(mainwin_view_menu_entries,
|
|
3160 mainwin_view_menu_entries_num,
|
|
3161 mainwin_accel);
|
|
3162
|
|
3163 mainwin_songname_menu = create_menu(mainwin_songname_menu_entries,
|
|
3164 mainwin_songname_menu_entries_num,
|
|
3165 mainwin_accel);
|
|
3166
|
|
3167 mainwin_add_menu = create_menu(mainwin_add_menu_entries,
|
|
3168 mainwin_add_menu_entries_num,
|
|
3169 mainwin_accel);
|
|
3170
|
|
3171 mainwin_vis_menu = create_menu(mainwin_vis_menu_entries,
|
|
3172 mainwin_vis_menu_entries_num,
|
|
3173 mainwin_accel);
|
|
3174
|
|
3175 make_submenu(mainwin_general_menu, "/View", mainwin_view_menu);
|
|
3176
|
|
3177 gtk_window_add_accel_group(GTK_WINDOW(mainwin), mainwin_accel);
|
|
3178 }
|
|
3179
|
|
3180 void
|
|
3181 mainwin_create(void)
|
|
3182 {
|
|
3183 mainwin_create_window();
|
|
3184
|
|
3185 mainwin_accel = gtk_accel_group_new();
|
|
3186 mainwin_create_menus();
|
|
3187
|
|
3188 mainwin_gc = gdk_gc_new(mainwin->window);
|
|
3189 mainwin_bg = gdk_pixmap_new(mainwin->window,
|
|
3190 MAINWIN_WIDTH, MAINWIN_HEIGHT, -1);
|
|
3191 mainwin_set_back_pixmap();
|
|
3192 mainwin_create_widgets();
|
|
3193
|
|
3194 vis_set_window(mainwin_vis, mainwin->window);
|
|
3195 }
|
|
3196
|
|
3197 void
|
|
3198 mainwin_attach_idle_func(void)
|
|
3199 {
|
|
3200 mainwin_timeout_id = g_timeout_add(MAINWIN_UPDATE_INTERVAL,
|
|
3201 mainwin_idle_func, NULL);
|
|
3202 }
|
|
3203
|
|
3204 static gboolean
|
|
3205 idle_func_change_song(gboolean waiting)
|
|
3206 {
|
|
3207 static GTimer *pause_timer = NULL;
|
|
3208
|
|
3209 if (!pause_timer)
|
|
3210 pause_timer = g_timer_new();
|
|
3211
|
|
3212 if (cfg.pause_between_songs) {
|
|
3213 gint timeleft;
|
|
3214
|
|
3215 if (!waiting) {
|
|
3216 g_timer_start(pause_timer);
|
|
3217 waiting = TRUE;
|
|
3218 }
|
|
3219
|
|
3220 timeleft = cfg.pause_between_songs_time -
|
|
3221 (gint) g_timer_elapsed(pause_timer, NULL);
|
|
3222
|
|
3223 number_set_number(mainwin_10min_num, timeleft / 600);
|
|
3224 number_set_number(mainwin_min_num, (timeleft / 60) % 10);
|
|
3225 number_set_number(mainwin_10sec_num, (timeleft / 10) % 6);
|
|
3226 number_set_number(mainwin_sec_num, timeleft % 10);
|
|
3227
|
|
3228 if (!mainwin_sposition->hs_pressed) {
|
|
3229 gchar time_str[5];
|
|
3230
|
|
3231 g_snprintf(time_str, sizeof(time_str), "%2.2d", timeleft / 60);
|
|
3232 textbox_set_text(mainwin_stime_min, time_str);
|
|
3233
|
|
3234 g_snprintf(time_str, sizeof(time_str), "%2.2d", timeleft % 60);
|
|
3235 textbox_set_text(mainwin_stime_sec, time_str);
|
|
3236 }
|
|
3237
|
|
3238 playlistwin_set_time(timeleft * 1000, 0, TIMER_ELAPSED);
|
|
3239 }
|
|
3240
|
|
3241 if (!cfg.pause_between_songs ||
|
|
3242 g_timer_elapsed(pause_timer, NULL) >= cfg.pause_between_songs_time) {
|
|
3243
|
|
3244 GDK_THREADS_ENTER();
|
|
3245 playlist_eof_reached();
|
|
3246 GDK_THREADS_LEAVE();
|
|
3247
|
|
3248 waiting = FALSE;
|
|
3249 }
|
|
3250
|
|
3251 return waiting;
|
|
3252 }
|
|
3253
|
|
3254 static void
|
|
3255 idle_func_update_song_info(gint time)
|
|
3256 {
|
|
3257 gint length, t;
|
|
3258 gchar stime_prefix;
|
|
3259
|
|
3260 length = playlist_get_current_length();
|
|
3261 playlistwin_set_time(time, length, cfg.timer_mode);
|
|
3262 input_update_vis(time);
|
|
3263
|
|
3264 if (cfg.timer_mode == TIMER_REMAINING) {
|
|
3265 if (length != -1) {
|
|
3266 number_set_number(mainwin_minus_num, 11);
|
|
3267 t = length - time;
|
|
3268 stime_prefix = '-';
|
|
3269 }
|
|
3270 else {
|
|
3271 number_set_number(mainwin_minus_num, 10);
|
|
3272 t = time;
|
|
3273 stime_prefix = ' ';
|
|
3274 }
|
|
3275 }
|
|
3276 else {
|
|
3277 number_set_number(mainwin_minus_num, 10);
|
|
3278 t = time;
|
|
3279 stime_prefix = ' ';
|
|
3280 }
|
|
3281 t /= 1000;
|
|
3282
|
|
3283 /* Show the time in the format HH:MM when we have more than 100
|
|
3284 * minutes. */
|
|
3285 if (t >= 100 * 60)
|
|
3286 t /= 60;
|
|
3287 number_set_number(mainwin_10min_num, t / 600);
|
|
3288 number_set_number(mainwin_min_num, (t / 60) % 10);
|
|
3289 number_set_number(mainwin_10sec_num, (t / 10) % 6);
|
|
3290 number_set_number(mainwin_sec_num, t % 10);
|
|
3291
|
|
3292 if (!mainwin_sposition->hs_pressed) {
|
|
3293 gchar *time_str;
|
|
3294
|
|
3295 time_str = g_strdup_printf("%c%2.2d", stime_prefix, t / 60);
|
|
3296 textbox_set_text(mainwin_stime_min, time_str);
|
|
3297 g_free(time_str);
|
|
3298
|
|
3299 time_str = g_strdup_printf("%2.2d", t % 60);
|
|
3300 textbox_set_text(mainwin_stime_sec, time_str);
|
|
3301 g_free(time_str);
|
|
3302 }
|
|
3303
|
|
3304 time /= 1000;
|
|
3305 length /= 1000;
|
|
3306 if (length > 0) {
|
|
3307 if (time > length) {
|
|
3308 hslider_set_position(mainwin_position, 219);
|
|
3309 hslider_set_position(mainwin_sposition, 13);
|
|
3310 }
|
|
3311 else {
|
|
3312 hslider_set_position(mainwin_position, (time * 219) / length);
|
|
3313 hslider_set_position(mainwin_sposition,
|
|
3314 ((time * 12) / length) + 1);
|
|
3315 }
|
|
3316 }
|
|
3317 else {
|
|
3318 hslider_set_position(mainwin_position, 0);
|
|
3319 hslider_set_position(mainwin_sposition, 1);
|
|
3320 }
|
|
3321 }
|
|
3322
|
|
3323
|
|
3324 static gboolean
|
|
3325 mainwin_idle_func(gpointer data)
|
|
3326 {
|
|
3327 static gboolean waiting = FALSE;
|
|
3328 static gint count = 0;
|
|
3329
|
|
3330 gint time;
|
|
3331
|
|
3332 if (bmp_playback_get_playing()) {
|
|
3333 GDK_THREADS_ENTER();
|
|
3334 vis_playback_start();
|
|
3335 GDK_THREADS_LEAVE();
|
|
3336
|
|
3337 time = bmp_playback_get_time();
|
|
3338
|
|
3339 switch (time) {
|
|
3340 case -1:
|
|
3341 /* no song playing */
|
|
3342 waiting = idle_func_change_song(waiting);
|
|
3343 break;
|
|
3344
|
|
3345 case -2:
|
|
3346 /* no usable output device */
|
|
3347 GDK_THREADS_ENTER();
|
|
3348 run_no_output_device_dialog();
|
|
3349 mainwin_stop_pushed();
|
|
3350 GDK_THREADS_LEAVE();
|
|
3351 waiting = FALSE;
|
|
3352 break;
|
|
3353
|
|
3354 default:
|
|
3355 /* song playing, all's well */
|
|
3356 idle_func_update_song_info(time);
|
|
3357 waiting = FALSE;
|
|
3358 }
|
|
3359 }
|
|
3360 else {
|
|
3361 GDK_THREADS_ENTER();
|
|
3362 vis_playback_stop();
|
|
3363 GDK_THREADS_LEAVE();
|
|
3364 }
|
|
3365
|
|
3366 GDK_THREADS_ENTER();
|
|
3367
|
|
3368 ctrlsocket_check();
|
|
3369
|
|
3370 draw_main_window(mainwin_force_redraw);
|
|
3371
|
|
3372 if (!count) {
|
|
3373 read_volume(VOLSET_UPDATE);
|
|
3374 count = 10;
|
|
3375 }
|
|
3376 else
|
|
3377 count--;
|
|
3378
|
|
3379 mainwin_force_redraw = FALSE;
|
|
3380 draw_playlist_window(FALSE);
|
|
3381 draw_equalizer_window(FALSE);
|
|
3382
|
|
3383 if (mainwin_title_text) {
|
|
3384 G_LOCK(mainwin_title);
|
|
3385 gtk_window_set_title(GTK_WINDOW(mainwin), mainwin_title_text);
|
|
3386 g_free(mainwin_title_text);
|
|
3387 mainwin_title_text = NULL;
|
|
3388 G_UNLOCK(mainwin_title);
|
|
3389
|
|
3390 mainwin_set_info_text();
|
|
3391 playlistwin_update_list();
|
|
3392 }
|
|
3393
|
|
3394 GDK_THREADS_LEAVE();
|
|
3395
|
|
3396 return TRUE;
|
|
3397 }
|