Mercurial > audlegacy
annotate src/audacious/ui_playlist.c @ 2488:c5075a79f1aa trunk
[svn] make input->set_info overwrite tuple->track_name. it allows input
plugins to update fileinfopopup along with meta data in a stream. it
may incur side effects, please let me know if you find any problem.
author | yaz |
---|---|
date | Thu, 08 Feb 2007 07:00:02 -0800 |
parents | d6315e592e40 |
children | 2aeea41ef023 |
rev | line source |
---|---|
2313 | 1 /* Audacious - Cross-platform multimedia player |
2 * Copyright (C) 2005-2006 Audacious development team. | |
3 * | |
4 * BMP - Cross-platform multimedia player | |
5 * Copyright (C) 2003-2004 BMP development team. | |
6 * | |
7 * Based on XMMS: | |
8 * Copyright (C) 1998-2003 XMMS development team. | |
9 * | |
10 * This program is free software; you can redistribute it and/or modify | |
11 * it under the terms of the GNU General Public License as published by | |
12 * the Free Software Foundation; under version 2 of the License. | |
13 * | |
14 * This program is distributed in the hope that it will be useful, | |
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
17 * GNU General Public License for more details. | |
18 * | |
19 * You should have received a copy of the GNU General Public License | |
20 * along with this program; if not, write to the Free Software | |
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | |
22 */ | |
23 | |
24 #include "ui_playlist.h" | |
25 | |
26 #include <glib.h> | |
27 #include <glib/gi18n.h> | |
28 #include <gdk/gdk.h> | |
29 #include <gdk/gdkkeysyms.h> | |
30 #include <gtk/gtk.h> | |
31 #include <string.h> | |
32 | |
33 #include "platform/smartinclude.h" | |
34 | |
35 #include <unistd.h> | |
36 #include <errno.h> | |
37 | |
2416
0fd7f4f969ad
[svn] integrated urldecode.* from libaudacious into audacious directory, made separate ui_fileopener.*
mf0102
parents:
2373
diff
changeset
|
38 #include "actions-playlist.h" |
2313 | 39 #include "dnd.h" |
40 #include "dock.h" | |
41 #include "hints.h" | |
42 #include "input.h" | |
43 #include "main.h" | |
44 #include "playback.h" | |
45 #include "playlist.h" | |
46 #include "playlist_container.h" | |
47 #include "playlist_manager.h" | |
2373
ad1d7687814c
[svn] made strings.h for existing strings.c, cleanups
mf0102
parents:
2363
diff
changeset
|
48 #include "strings.h" |
2416
0fd7f4f969ad
[svn] integrated urldecode.* from libaudacious into audacious directory, made separate ui_fileopener.*
mf0102
parents:
2373
diff
changeset
|
49 #include "ui_equalizer.h" |
0fd7f4f969ad
[svn] integrated urldecode.* from libaudacious into audacious directory, made separate ui_fileopener.*
mf0102
parents:
2373
diff
changeset
|
50 #include "ui_fileopener.h" |
0fd7f4f969ad
[svn] integrated urldecode.* from libaudacious into audacious directory, made separate ui_fileopener.*
mf0102
parents:
2373
diff
changeset
|
51 #include "ui_fileinfopopup.h" |
0fd7f4f969ad
[svn] integrated urldecode.* from libaudacious into audacious directory, made separate ui_fileopener.*
mf0102
parents:
2373
diff
changeset
|
52 #include "ui_main.h" |
0fd7f4f969ad
[svn] integrated urldecode.* from libaudacious into audacious directory, made separate ui_fileopener.*
mf0102
parents:
2373
diff
changeset
|
53 #include "ui_manager.h" |
0fd7f4f969ad
[svn] integrated urldecode.* from libaudacious into audacious directory, made separate ui_fileopener.*
mf0102
parents:
2373
diff
changeset
|
54 #include "util.h" |
2313 | 55 |
56 #include "icons-stock.h" | |
57 #include "images/audacious_playlist.xpm" | |
58 | |
59 GtkWidget *playlistwin; | |
60 | |
61 PlayList_List *playlistwin_list = NULL; | |
62 PButton *playlistwin_shade, *playlistwin_close; | |
63 | |
64 static gboolean playlistwin_resizing = FALSE; | |
65 | |
66 static GdkPixmap *playlistwin_bg; | |
67 static GdkBitmap *playlistwin_mask = NULL; | |
68 static GdkGC *playlistwin_gc; | |
69 | |
70 static gboolean playlistwin_hint_flag = FALSE; | |
71 | |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
72 static GtkWidget *playlistwin_infopopup = NULL; |
2351
911743d27aba
[svn] - simplify and optimize the metadata tooltip trigger function
giacomo
parents:
2348
diff
changeset
|
73 static guint playlistwin_infopopup_sid = 0; |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
74 |
2313 | 75 static PlaylistSlider *playlistwin_slider = NULL; |
76 static TextBox *playlistwin_time_min, *playlistwin_time_sec; | |
77 static TextBox *playlistwin_info, *playlistwin_sinfo; | |
78 static SButton *playlistwin_srew, *playlistwin_splay; | |
79 static SButton *playlistwin_spause, *playlistwin_sstop; | |
80 static SButton *playlistwin_sfwd, *playlistwin_seject; | |
81 static SButton *playlistwin_sscroll_up, *playlistwin_sscroll_down; | |
82 | |
83 static GList *playlistwin_wlist = NULL; | |
84 | |
85 void playlistwin_select_search_cbt_cb( GtkWidget *called_cbt , | |
86 gpointer other_cbt ); | |
87 static gboolean playlistwin_select_search_kp_cb( GtkWidget *entry , GdkEventKey *event , | |
88 gpointer searchdlg_win ); | |
89 | |
90 static void playlistwin_draw_frame(void); | |
91 | |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
92 static gboolean playlistwin_fileinfopopup_probe(gpointer * filepopup_win); |
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
93 |
2313 | 94 |
95 gboolean | |
96 playlistwin_is_shaded(void) | |
97 { | |
98 return cfg.playlist_shaded; | |
99 } | |
100 | |
101 gint | |
102 playlistwin_get_width(void) | |
103 { | |
104 cfg.playlist_width /= PLAYLISTWIN_WIDTH_SNAP; | |
105 cfg.playlist_width *= PLAYLISTWIN_WIDTH_SNAP; | |
106 return cfg.playlist_width; | |
107 } | |
108 | |
109 gint | |
110 playlistwin_get_height_unshaded(void) | |
111 { | |
112 gint height; | |
113 cfg.playlist_height /= PLAYLISTWIN_HEIGHT_SNAP; | |
114 cfg.playlist_height *= PLAYLISTWIN_HEIGHT_SNAP; | |
115 height = cfg.playlist_height; | |
116 return height; | |
117 } | |
118 | |
119 gint | |
120 playlistwin_get_height_shaded(void) | |
121 { | |
122 return PLAYLISTWIN_SHADED_HEIGHT; | |
123 } | |
124 | |
125 gint | |
126 playlistwin_get_height(void) | |
127 { | |
128 if (playlistwin_is_shaded()) | |
129 return playlistwin_get_height_shaded(); | |
130 else | |
131 return playlistwin_get_height_unshaded(); | |
132 } | |
133 | |
134 void | |
135 playlistwin_get_size(gint * width, gint * height) | |
136 { | |
137 if (width) | |
138 *width = playlistwin_get_width(); | |
139 | |
140 if (height) | |
141 *height = playlistwin_get_height(); | |
142 } | |
143 | |
144 static void | |
145 playlistwin_update_info(Playlist *playlist) | |
146 { | |
147 gchar *text, *sel_text, *tot_text; | |
148 gulong selection, total; | |
149 gboolean selection_more, total_more; | |
150 | |
151 playlist_get_total_time(playlist, &total, &selection, &total_more, &selection_more); | |
152 | |
153 if (selection > 0 || (selection == 0 && !selection_more)) { | |
154 if (selection > 3600) | |
155 sel_text = | |
156 g_strdup_printf("%lu:%-2.2lu:%-2.2lu%s", selection / 3600, | |
157 (selection / 60) % 60, selection % 60, | |
158 (selection_more ? "+" : "")); | |
159 else | |
160 sel_text = | |
161 g_strdup_printf("%lu:%-2.2lu%s", selection / 60, | |
162 selection % 60, (selection_more ? "+" : "")); | |
163 } | |
164 else | |
165 sel_text = g_strdup("?"); | |
166 if (total > 0 || (total == 0 && !total_more)) { | |
167 if (total > 3600) | |
168 tot_text = | |
169 g_strdup_printf("%lu:%-2.2lu:%-2.2lu%s", total / 3600, | |
170 (total / 60) % 60, total % 60, | |
171 total_more ? "+" : ""); | |
172 else | |
173 tot_text = | |
174 g_strdup_printf("%lu:%-2.2lu%s", total / 60, total % 60, | |
175 total_more ? "+" : ""); | |
176 } | |
177 else | |
178 tot_text = g_strdup("?"); | |
179 text = g_strconcat(sel_text, "/", tot_text, NULL); | |
180 textbox_set_text(playlistwin_info, text ? text : ""); | |
181 g_free(text); | |
182 g_free(tot_text); | |
183 g_free(sel_text); | |
184 } | |
185 | |
186 static void | |
187 playlistwin_update_sinfo(Playlist *playlist) | |
188 { | |
189 gchar *posstr, *timestr, *title, *info; | |
190 gint pos, time; | |
191 | |
192 pos = playlist_get_position(playlist); | |
193 title = playlist_get_songtitle(playlist, pos); | |
194 | |
195 if (!title) { | |
196 textbox_set_text(playlistwin_sinfo, ""); | |
197 return; | |
198 } | |
199 | |
200 convert_title_text(title); | |
201 | |
202 time = playlist_get_songtime(playlist, pos); | |
203 | |
204 if (cfg.show_numbers_in_pl) | |
205 posstr = g_strdup_printf("%d. ", pos + 1); | |
206 else | |
207 posstr = g_strdup(""); | |
208 | |
209 if (time != -1) { | |
210 timestr = g_strdup_printf(" (%d:%-2.2d)", time / 60000, | |
211 (time / 1000) % 60); | |
212 } | |
213 else | |
214 timestr = g_strdup(""); | |
215 | |
216 info = g_strdup_printf("%s%s%s", posstr, title, timestr); | |
217 | |
218 g_free(posstr); | |
219 g_free(title); | |
220 g_free(timestr); | |
221 | |
222 textbox_set_text(playlistwin_sinfo, info ? info : ""); | |
223 g_free(info); | |
224 } | |
225 | |
226 gboolean | |
227 playlistwin_item_visible(gint index) | |
228 { | |
229 if (index >= playlistwin_list->pl_first | |
230 && index < | |
231 (playlistwin_list->pl_first + playlistwin_list->pl_num_visible)) | |
232 return TRUE; | |
233 return FALSE; | |
234 } | |
235 | |
236 gint | |
237 playlistwin_get_toprow(void) | |
238 { | |
239 if (playlistwin_list) | |
240 return (playlistwin_list->pl_first); | |
241 return (-1); | |
242 } | |
243 | |
244 void | |
245 playlistwin_set_toprow(gint toprow) | |
246 { | |
247 if (playlistwin_list) | |
248 playlistwin_list->pl_first = toprow; | |
249 playlistwin_update_list(playlist_get_active()); | |
250 } | |
251 | |
252 void | |
253 playlistwin_update_list(Playlist *playlist) | |
254 { | |
255 /* this can happen early on. just bail gracefully. */ | |
2363 | 256 g_return_if_fail(playlistwin_list); |
2313 | 257 |
258 widget_draw(WIDGET(playlistwin_list)); | |
259 widget_draw(WIDGET(playlistwin_slider)); | |
260 playlistwin_update_info(playlist); | |
261 playlistwin_update_sinfo(playlist); | |
262 } | |
263 | |
264 static void | |
265 playlistwin_set_mask(void) | |
266 { | |
267 GdkGC *gc; | |
268 GdkColor pattern; | |
269 | |
270 if (playlistwin_mask) | |
271 g_object_unref(playlistwin_mask); | |
272 | |
273 playlistwin_mask = | |
274 gdk_pixmap_new(playlistwin->window, playlistwin_get_width(), | |
275 playlistwin_get_height(), 1); | |
276 gc = gdk_gc_new(playlistwin_mask); | |
277 pattern.pixel = 1; | |
278 gdk_gc_set_foreground(gc, &pattern); | |
279 gdk_draw_rectangle(playlistwin_mask, gc, TRUE, 0, 0, | |
280 playlistwin_get_width(), playlistwin_get_height()); | |
281 g_object_unref(gc); | |
282 | |
283 gtk_widget_shape_combine_mask(playlistwin, playlistwin_mask, 0, 0); | |
284 } | |
285 | |
286 static void | |
287 playlistwin_set_geometry_hints(gboolean shaded) | |
288 { | |
289 GdkGeometry geometry; | |
290 GdkWindowHints mask; | |
291 | |
292 geometry.min_width = PLAYLISTWIN_MIN_WIDTH; | |
293 geometry.max_width = G_MAXUINT16; | |
294 | |
295 geometry.width_inc = PLAYLISTWIN_WIDTH_SNAP; | |
296 geometry.height_inc = PLAYLISTWIN_HEIGHT_SNAP; | |
297 | |
298 if (shaded) { | |
299 geometry.min_height = PLAYLISTWIN_SHADED_HEIGHT; | |
300 geometry.max_height = PLAYLISTWIN_SHADED_HEIGHT; | |
301 geometry.base_height = PLAYLISTWIN_SHADED_HEIGHT; | |
302 } | |
303 else { | |
304 geometry.min_height = PLAYLISTWIN_MIN_HEIGHT; | |
305 geometry.max_height = G_MAXUINT16; | |
306 } | |
307 | |
308 mask = GDK_HINT_MIN_SIZE | GDK_HINT_MAX_SIZE | GDK_HINT_RESIZE_INC; | |
309 | |
310 gtk_window_set_geometry_hints(GTK_WINDOW(playlistwin), | |
311 playlistwin, &geometry, mask); | |
312 } | |
313 | |
314 void | |
315 playlistwin_set_sinfo_font(gchar *font) | |
316 { | |
317 gchar *tmp = NULL, *tmp2 = NULL; | |
318 | |
2363 | 319 g_return_if_fail(font); |
2313 | 320 |
321 tmp = g_strdup(font); | |
2363 | 322 g_return_if_fail(tmp); |
2313 | 323 |
324 *strrchr(tmp, ' ') = '\0'; | |
325 tmp2 = g_strdup_printf("%s 8", tmp); | |
2363 | 326 g_return_if_fail(tmp2); |
327 | |
2313 | 328 textbox_set_xfont(playlistwin_sinfo, cfg.mainwin_use_xfont, tmp2); |
329 | |
2363 | 330 g_free(tmp); |
331 g_free(tmp2); | |
2313 | 332 } |
333 | |
334 void | |
335 playlistwin_set_sinfo_scroll(gboolean scroll) | |
336 { | |
337 if(playlistwin_is_shaded()) | |
338 textbox_set_scroll(playlistwin_sinfo, cfg.autoscroll); | |
339 else | |
340 textbox_set_scroll(playlistwin_sinfo, FALSE); | |
341 } | |
342 | |
343 void | |
344 playlistwin_set_shade(gboolean shaded) | |
345 { | |
346 cfg.playlist_shaded = shaded; | |
347 | |
348 if (shaded) { | |
349 playlistwin_set_sinfo_font(cfg.playlist_font); | |
350 playlistwin_set_sinfo_scroll(cfg.autoscroll); | |
351 widget_show(WIDGET(playlistwin_sinfo)); | |
352 playlistwin_shade->pb_nx = 128; | |
353 playlistwin_shade->pb_ny = 45; | |
354 playlistwin_shade->pb_px = 150; | |
355 playlistwin_shade->pb_py = 42; | |
356 playlistwin_close->pb_nx = 138; | |
357 playlistwin_close->pb_ny = 45; | |
358 } | |
359 else { | |
360 widget_hide(WIDGET(playlistwin_sinfo)); | |
361 playlistwin_set_sinfo_scroll(FALSE); | |
362 playlistwin_shade->pb_nx = 157; | |
363 playlistwin_shade->pb_ny = 3; | |
364 playlistwin_shade->pb_px = 62; | |
365 playlistwin_shade->pb_py = 42; | |
366 playlistwin_close->pb_nx = 167; | |
367 playlistwin_close->pb_ny = 3; | |
368 } | |
369 | |
370 dock_shade(dock_window_list, GTK_WINDOW(playlistwin), | |
371 playlistwin_get_height()); | |
372 | |
373 playlistwin_set_geometry_hints(cfg.playlist_shaded); | |
374 | |
375 gtk_window_resize(GTK_WINDOW(playlistwin), | |
376 playlistwin_get_width(), | |
377 playlistwin_get_height()); | |
378 | |
379 playlistwin_set_mask(); | |
380 | |
381 widget_draw(WIDGET(playlistwin_list)); | |
382 widget_draw(WIDGET(playlistwin_slider)); | |
383 | |
384 draw_playlist_window(TRUE); | |
385 } | |
386 | |
387 static void | |
388 playlistwin_set_shade_menu(gboolean shaded) | |
389 { | |
390 GtkAction *action = gtk_action_group_get_action( | |
391 toggleaction_group_others , "roll up playlist editor" ); | |
392 gtk_toggle_action_set_active( GTK_TOGGLE_ACTION(action) , shaded ); | |
393 | |
394 playlistwin_set_shade(shaded); | |
395 playlistwin_update_list(playlist_get_active()); | |
396 } | |
397 | |
398 void | |
399 playlistwin_shade_toggle(void) | |
400 { | |
401 playlistwin_set_shade_menu(!cfg.playlist_shaded); | |
402 } | |
403 | |
404 static void | |
405 playlistwin_release(GtkWidget * widget, | |
406 GdkEventButton * event, | |
407 gpointer callback_data) | |
408 { | |
409 if (event->button == 3) | |
410 return; | |
411 | |
412 gdk_pointer_ungrab(GDK_CURRENT_TIME); | |
413 playlistwin_resizing = FALSE; | |
414 gdk_flush(); | |
415 | |
416 if (dock_is_moving(GTK_WINDOW(playlistwin))) | |
417 dock_move_release(GTK_WINDOW(playlistwin)); | |
418 else | |
419 { | |
420 handle_release_cb(playlistwin_wlist, widget, event); | |
421 draw_playlist_window(FALSE); | |
422 } | |
423 } | |
424 | |
425 void | |
426 playlistwin_scroll(gint num) | |
427 { | |
428 playlistwin_list->pl_first += num; | |
429 playlistwin_update_list(playlist_get_active()); | |
430 } | |
431 | |
432 void | |
433 playlistwin_scroll_up_pushed(void) | |
434 { | |
435 playlistwin_scroll(-3); | |
436 } | |
437 | |
438 void | |
439 playlistwin_scroll_down_pushed(void) | |
440 { | |
441 playlistwin_scroll(3); | |
442 } | |
443 | |
444 static void | |
445 playlistwin_select_all(void) | |
446 { | |
447 Playlist *playlist = playlist_get_active(); | |
448 | |
449 playlist_select_all(playlist, TRUE); | |
450 playlistwin_list->pl_prev_selected = 0; | |
451 playlistwin_list->pl_prev_min = 0; | |
452 playlistwin_list->pl_prev_max = playlist_get_length(playlist) - 1; | |
453 playlistwin_update_list(playlist); | |
454 } | |
455 | |
456 static void | |
457 playlistwin_select_none(void) | |
458 { | |
459 playlist_select_all(playlist_get_active(), FALSE); | |
460 playlistwin_list->pl_prev_selected = -1; | |
461 playlistwin_list->pl_prev_min = -1; | |
462 playlistwin_update_list(playlist_get_active()); | |
463 } | |
464 | |
465 static void | |
466 playlistwin_select_search(void) | |
467 { | |
468 Playlist *playlist = playlist_get_active(); | |
469 GtkWidget *searchdlg_win, *searchdlg_table; | |
470 GtkWidget *searchdlg_hbox, *searchdlg_logo, *searchdlg_helptext; | |
471 GtkWidget *searchdlg_entry_track_name, *searchdlg_label_track_name; | |
472 GtkWidget *searchdlg_entry_album_name, *searchdlg_label_album_name; | |
473 GtkWidget *searchdlg_entry_file_name, *searchdlg_label_file_name; | |
474 GtkWidget *searchdlg_entry_performer, *searchdlg_label_performer; | |
475 GtkWidget *searchdlg_checkbt_clearprevsel; | |
476 GtkWidget *searchdlg_checkbt_newplaylist; | |
477 GtkWidget *searchdlg_checkbt_autoenqueue; | |
478 gint result; | |
479 | |
480 /* create dialog */ | |
481 searchdlg_win = gtk_dialog_new_with_buttons( | |
482 _("Search entries in active playlist") , GTK_WINDOW(mainwin) , | |
483 GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT , | |
484 GTK_STOCK_CANCEL , GTK_RESPONSE_REJECT , GTK_STOCK_OK , GTK_RESPONSE_ACCEPT , NULL ); | |
485 /* help text and logo */ | |
486 searchdlg_hbox = gtk_hbox_new( FALSE , 4 ); | |
487 searchdlg_logo = gtk_image_new_from_stock( GTK_STOCK_FIND , GTK_ICON_SIZE_DIALOG ); | |
488 searchdlg_helptext = gtk_label_new( _("Select entries in playlist by filling one or more " | |
489 "fields. Fields use regular expressions syntax, case-insensitive. If you don't know how " | |
490 "regular expressions work, simply insert a literal portion of what you're searching for.") ); | |
491 gtk_label_set_line_wrap( GTK_LABEL(searchdlg_helptext) , TRUE ); | |
492 gtk_box_pack_start( GTK_BOX(searchdlg_hbox) , searchdlg_logo , FALSE , FALSE , 0 ); | |
493 gtk_box_pack_start( GTK_BOX(searchdlg_hbox) , searchdlg_helptext , FALSE , FALSE , 0 ); | |
494 /* track name */ | |
495 searchdlg_label_track_name = gtk_label_new( _("Track name: ") ); | |
496 searchdlg_entry_track_name = gtk_entry_new(); | |
497 gtk_misc_set_alignment( GTK_MISC(searchdlg_label_track_name) , 0 , 0.5 ); | |
498 g_signal_connect( G_OBJECT(searchdlg_entry_track_name) , "key-press-event" , | |
499 G_CALLBACK(playlistwin_select_search_kp_cb) , searchdlg_win ); | |
500 /* album name */ | |
501 searchdlg_label_album_name = gtk_label_new( _("Album name: ") ); | |
502 searchdlg_entry_album_name = gtk_entry_new(); | |
503 gtk_misc_set_alignment( GTK_MISC(searchdlg_label_album_name) , 0 , 0.5 ); | |
504 g_signal_connect( G_OBJECT(searchdlg_entry_album_name) , "key-press-event" , | |
505 G_CALLBACK(playlistwin_select_search_kp_cb) , searchdlg_win ); | |
506 /* artist */ | |
507 searchdlg_label_performer = gtk_label_new( _("Artist: ") ); | |
508 searchdlg_entry_performer = gtk_entry_new(); | |
509 gtk_misc_set_alignment( GTK_MISC(searchdlg_label_performer) , 0 , 0.5 ); | |
510 g_signal_connect( G_OBJECT(searchdlg_entry_performer) , "key-press-event" , | |
511 G_CALLBACK(playlistwin_select_search_kp_cb) , searchdlg_win ); | |
512 /* file name */ | |
513 searchdlg_label_file_name = gtk_label_new( _("Filename: ") ); | |
514 searchdlg_entry_file_name = gtk_entry_new(); | |
515 gtk_misc_set_alignment( GTK_MISC(searchdlg_label_file_name) , 0 , 0.5 ); | |
516 g_signal_connect( G_OBJECT(searchdlg_entry_file_name) , "key-press-event" , | |
517 G_CALLBACK(playlistwin_select_search_kp_cb) , searchdlg_win ); | |
518 /* some options that control behaviour */ | |
519 searchdlg_checkbt_clearprevsel = gtk_check_button_new_with_label( | |
520 _("Clear previous selection before searching") ); | |
521 gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(searchdlg_checkbt_clearprevsel) , TRUE ); | |
522 searchdlg_checkbt_autoenqueue = gtk_check_button_new_with_label( | |
523 _("Automatically toggle queue for matching entries") ); | |
524 gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(searchdlg_checkbt_autoenqueue) , FALSE ); | |
525 searchdlg_checkbt_newplaylist = gtk_check_button_new_with_label( | |
526 _("Create a new playlist with matching entries") ); | |
527 gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(searchdlg_checkbt_newplaylist) , FALSE ); | |
528 g_signal_connect( G_OBJECT(searchdlg_checkbt_autoenqueue) , "clicked" , | |
529 G_CALLBACK(playlistwin_select_search_cbt_cb) , searchdlg_checkbt_newplaylist ); | |
530 g_signal_connect( G_OBJECT(searchdlg_checkbt_newplaylist) , "clicked" , | |
531 G_CALLBACK(playlistwin_select_search_cbt_cb) , searchdlg_checkbt_autoenqueue ); | |
532 /* place fields in searchdlg_table */ | |
533 searchdlg_table = gtk_table_new( 8 , 2 , FALSE ); | |
534 gtk_table_set_row_spacing( GTK_TABLE(searchdlg_table) , 0 , 8 ); | |
535 gtk_table_set_row_spacing( GTK_TABLE(searchdlg_table) , 4 , 8 ); | |
536 gtk_table_attach( GTK_TABLE(searchdlg_table) , searchdlg_hbox , | |
537 0 , 2 , 0 , 1 , GTK_FILL | GTK_EXPAND , GTK_FILL | GTK_EXPAND , 0 , 2 ); | |
538 gtk_table_attach( GTK_TABLE(searchdlg_table) , searchdlg_label_track_name , | |
539 0 , 1 , 1 , 2 , GTK_FILL , GTK_FILL | GTK_EXPAND , 0 , 2 ); | |
540 gtk_table_attach( GTK_TABLE(searchdlg_table) , searchdlg_entry_track_name , | |
541 1 , 2 , 1 , 2 , GTK_FILL | GTK_EXPAND , GTK_FILL | GTK_EXPAND , 0 , 2 ); | |
542 gtk_table_attach( GTK_TABLE(searchdlg_table) , searchdlg_label_album_name , | |
543 0 , 1 , 2 , 3 , GTK_FILL , GTK_FILL | GTK_EXPAND , 0 , 2 ); | |
544 gtk_table_attach( GTK_TABLE(searchdlg_table) , searchdlg_entry_album_name , | |
545 1 , 2 , 2 , 3 , GTK_FILL | GTK_EXPAND , GTK_FILL | GTK_EXPAND , 0 , 2 ); | |
546 gtk_table_attach( GTK_TABLE(searchdlg_table) , searchdlg_label_performer , | |
547 0 , 1 , 3 , 4 , GTK_FILL , GTK_FILL | GTK_EXPAND , 0 , 2 ); | |
548 gtk_table_attach( GTK_TABLE(searchdlg_table) , searchdlg_entry_performer , | |
549 1 , 2 , 3 , 4 , GTK_FILL | GTK_EXPAND , GTK_FILL | GTK_EXPAND , 0 , 2 ); | |
550 gtk_table_attach( GTK_TABLE(searchdlg_table) , searchdlg_label_file_name , | |
551 0 , 1 , 4 , 5 , GTK_FILL , GTK_FILL | GTK_EXPAND , 0 , 2 ); | |
552 gtk_table_attach( GTK_TABLE(searchdlg_table) , searchdlg_entry_file_name , | |
553 1 , 2 , 4 , 5 , GTK_FILL | GTK_EXPAND , GTK_FILL | GTK_EXPAND , 0 , 2 ); | |
554 gtk_table_attach( GTK_TABLE(searchdlg_table) , searchdlg_checkbt_clearprevsel , | |
555 0 , 2 , 5 , 6 , GTK_FILL | GTK_EXPAND , GTK_FILL | GTK_EXPAND , 0 , 1 ); | |
556 gtk_table_attach( GTK_TABLE(searchdlg_table) , searchdlg_checkbt_autoenqueue , | |
557 0 , 2 , 6 , 7 , GTK_FILL | GTK_EXPAND , GTK_FILL | GTK_EXPAND , 0 , 1 ); | |
558 gtk_table_attach( GTK_TABLE(searchdlg_table) , searchdlg_checkbt_newplaylist , | |
559 0 , 2 , 7 , 8 , GTK_FILL | GTK_EXPAND , GTK_FILL | GTK_EXPAND , 0 , 1 ); | |
560 | |
561 gtk_container_set_border_width( GTK_CONTAINER(searchdlg_table) , 5 ); | |
562 gtk_container_add( GTK_CONTAINER(GTK_DIALOG(searchdlg_win)->vbox) , searchdlg_table ); | |
563 gtk_widget_show_all( searchdlg_win ); | |
564 result = gtk_dialog_run( GTK_DIALOG(searchdlg_win) ); | |
565 switch(result) | |
566 { | |
567 case GTK_RESPONSE_ACCEPT: | |
568 { | |
569 gint matched_entries_num = 0; | |
570 /* create a TitleInput tuple with user search data */ | |
571 TitleInput *tuple = g_malloc(sizeof(TitleInput)); | |
572 gchar *searchdata = NULL; | |
573 searchdata = (gchar*)gtk_entry_get_text( GTK_ENTRY(searchdlg_entry_track_name) ); | |
574 tuple->track_name = ( strcmp(searchdata,"") ) ? g_strdup(searchdata) : NULL; | |
575 searchdata = (gchar*)gtk_entry_get_text( GTK_ENTRY(searchdlg_entry_album_name) ); | |
576 tuple->album_name = ( strcmp(searchdata,"") ) ? g_strdup(searchdata) : NULL; | |
577 searchdata = (gchar*)gtk_entry_get_text( GTK_ENTRY(searchdlg_entry_performer) ); | |
578 tuple->performer = ( strcmp(searchdata,"") ) ? g_strdup(searchdata) : NULL; | |
579 searchdata = (gchar*)gtk_entry_get_text( GTK_ENTRY(searchdlg_entry_file_name) ); | |
580 tuple->file_name = ( strcmp(searchdata,"") ) ? g_strdup(searchdata) : NULL; | |
581 /* check if previous selection should be cleared before searching */ | |
582 if ( gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(searchdlg_checkbt_clearprevsel)) == TRUE ) | |
583 playlistwin_select_none(); | |
584 /* now send this tuple to the real search function */ | |
585 matched_entries_num = playlist_select_search( playlist , tuple , 0 ); | |
586 /* we do not need the tuple and its data anymore */ | |
587 if ( tuple->track_name != NULL ) g_free( tuple->track_name ); | |
588 if ( tuple->album_name != NULL ) g_free( tuple->album_name ); | |
589 if ( tuple->performer != NULL ) g_free( tuple->performer ); | |
590 if ( tuple->file_name != NULL ) g_free( tuple->file_name ); | |
591 g_free( tuple ); | |
592 playlistwin_update_list(playlist_get_active()); | |
593 /* check if a new playlist should be created after searching */ | |
594 if ( gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(searchdlg_checkbt_newplaylist)) == TRUE ) | |
595 playlist_new_from_selected(); | |
596 /* check if matched entries should be queued */ | |
597 else if ( gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(searchdlg_checkbt_autoenqueue)) == TRUE ) | |
598 playlist_queue(playlist_get_active()); | |
599 break; | |
600 } | |
601 default: | |
602 break; | |
603 } | |
604 /* done here :) */ | |
605 gtk_widget_destroy( searchdlg_win ); | |
606 } | |
607 | |
608 static void | |
609 playlistwin_inverse_selection(void) | |
610 { | |
611 playlist_select_invert_all(playlist_get_active()); | |
612 playlistwin_list->pl_prev_selected = -1; | |
613 playlistwin_list->pl_prev_min = -1; | |
614 playlistwin_update_list(playlist_get_active()); | |
615 } | |
616 | |
617 static void | |
618 playlistwin_resize(gint width, gint height) | |
619 { | |
620 gboolean redraw; | |
621 | |
622 g_return_if_fail(width > 0 && height > 0); | |
623 | |
624 cfg.playlist_width = width; | |
625 | |
626 if (!cfg.playlist_shaded) | |
627 cfg.playlist_height = height; | |
628 else | |
629 height = cfg.playlist_height; | |
630 | |
631 /* FIXME: why the fsck are we doing this manually? */ | |
632 /* adjust widget positions and sizes */ | |
633 | |
634 widget_resize(WIDGET(playlistwin_list), width - 31, height - 58); | |
635 | |
636 widget_move(WIDGET(playlistwin_slider), width - 15, 20); | |
637 widget_resize(WIDGET(playlistwin_slider), 8, height - 58); | |
638 | |
639 widget_resize(WIDGET(playlistwin_sinfo), width - 35, 14); | |
640 playlistwin_update_sinfo(playlist_get_active()); | |
641 | |
642 widget_move(WIDGET(playlistwin_shade), width - 21, 3); | |
643 widget_move(WIDGET(playlistwin_close), width - 11, 3); | |
644 widget_move(WIDGET(playlistwin_time_min), width - 82, height - 15); | |
645 widget_move(WIDGET(playlistwin_time_sec), width - 64, height - 15); | |
646 widget_move(WIDGET(playlistwin_info), width - 143, height - 28); | |
647 widget_move(WIDGET(playlistwin_srew), width - 144, height - 16); | |
648 widget_move(WIDGET(playlistwin_splay), width - 138, height - 16); | |
649 widget_move(WIDGET(playlistwin_spause), width - 128, height - 16); | |
650 widget_move(WIDGET(playlistwin_sstop), width - 118, height - 16); | |
651 widget_move(WIDGET(playlistwin_sfwd), width - 109, height - 16); | |
652 widget_move(WIDGET(playlistwin_seject), width - 100, height - 16); | |
653 widget_move(WIDGET(playlistwin_sscroll_up), width - 14, height - 35); | |
654 widget_move(WIDGET(playlistwin_sscroll_down), width - 14, height - 30); | |
655 | |
656 g_object_unref(playlistwin_bg); | |
657 playlistwin_bg = gdk_pixmap_new(playlistwin->window, width, height, -1); | |
658 playlistwin_set_mask(); | |
659 | |
660 widget_list_lock(playlistwin_wlist); | |
661 | |
662 widget_list_change_pixmap(playlistwin_wlist, playlistwin_bg); | |
663 playlistwin_draw_frame(); | |
664 widget_list_draw(playlistwin_wlist, &redraw, TRUE); | |
665 widget_list_clear_redraw(playlistwin_wlist); | |
666 | |
667 widget_list_unlock(playlistwin_wlist); | |
668 | |
669 gdk_window_set_back_pixmap(playlistwin->window, playlistwin_bg, 0); | |
670 gdk_window_clear(playlistwin->window); | |
671 } | |
672 | |
673 | |
674 | |
675 static void | |
676 playlistwin_motion(GtkWidget * widget, | |
677 GdkEventMotion * event, | |
678 gpointer callback_data) | |
679 { | |
680 GdkEvent *gevent; | |
681 | |
682 if (dock_is_moving(GTK_WINDOW(playlistwin))) { | |
683 dock_move_motion(GTK_WINDOW(playlistwin), event); | |
684 } | |
685 else { | |
686 handle_motion_cb(playlistwin_wlist, widget, event); | |
687 draw_playlist_window(FALSE); | |
688 } | |
689 gdk_flush(); | |
690 | |
691 while ((gevent = gdk_event_get()) != NULL) gdk_event_free(gevent); | |
692 } | |
693 | |
694 static void | |
695 playlistwin_enter(GtkWidget * widget, | |
696 GdkEventMotion * event, | |
697 gpointer callback_data) | |
698 { | |
699 playlistwin_list->pl_tooltips = TRUE; | |
700 } | |
701 | |
702 static void | |
703 playlistwin_leave(GtkWidget * widget, | |
704 GdkEventMotion * event, | |
705 gpointer callback_data) | |
706 { | |
707 playlistwin_list->pl_tooltips = FALSE; | |
708 } | |
709 | |
710 static void | |
711 playlistwin_show_filebrowser(void) | |
712 { | |
2416
0fd7f4f969ad
[svn] integrated urldecode.* from libaudacious into audacious directory, made separate ui_fileopener.*
mf0102
parents:
2373
diff
changeset
|
713 run_filebrowser(NO_PLAY_BUTTON); |
2313 | 714 } |
715 | |
716 static void | |
717 playlistwin_fileinfo(void) | |
718 { | |
719 Playlist *playlist = playlist_get_active(); | |
720 | |
721 /* Show the first selected file, or the current file if nothing is | |
722 * selected */ | |
723 GList *list = playlist_get_selected(playlist); | |
724 if (list) { | |
725 playlist_fileinfo(playlist, GPOINTER_TO_INT(list->data)); | |
726 g_list_free(list); | |
727 } | |
728 else | |
729 playlist_fileinfo_current(playlist); | |
730 } | |
731 | |
732 static void | |
2363 | 733 show_playlist_save_error(GtkWindow *parent, |
734 const gchar *filename) | |
2313 | 735 { |
736 GtkWidget *dialog; | |
737 | |
738 g_return_if_fail(GTK_IS_WINDOW(parent)); | |
2363 | 739 g_return_if_fail(filename); |
2313 | 740 |
741 dialog = gtk_message_dialog_new(GTK_WINDOW(parent), | |
742 GTK_DIALOG_DESTROY_WITH_PARENT, | |
743 GTK_MESSAGE_ERROR, | |
744 GTK_BUTTONS_OK, | |
745 _("Error writing playlist \"%s\": %s"), | |
746 filename, strerror(errno)); | |
747 | |
748 gtk_dialog_run(GTK_DIALOG(dialog)); | |
749 gtk_widget_destroy(dialog); | |
750 } | |
751 | |
752 static gboolean | |
753 show_playlist_overwrite_prompt(GtkWindow * parent, | |
754 const gchar * filename) | |
755 { | |
756 GtkWidget *dialog; | |
757 gint result; | |
758 | |
759 g_return_val_if_fail(GTK_IS_WINDOW(parent), FALSE); | |
760 g_return_val_if_fail(filename != NULL, FALSE); | |
761 | |
762 dialog = gtk_message_dialog_new(GTK_WINDOW(parent), | |
763 GTK_DIALOG_DESTROY_WITH_PARENT, | |
764 GTK_MESSAGE_QUESTION, | |
765 GTK_BUTTONS_YES_NO, | |
766 _("%s already exist. Continue?"), | |
767 filename); | |
768 | |
769 result = gtk_dialog_run(GTK_DIALOG(dialog)); | |
770 gtk_widget_destroy(dialog); | |
771 | |
772 return (result == GTK_RESPONSE_YES); | |
773 } | |
774 | |
775 static void | |
776 show_playlist_save_format_error(GtkWindow * parent, | |
777 const gchar * filename) | |
778 { | |
779 const gchar *markup = | |
780 N_("<b><big>Unable to save playlist.</big></b>\n\n" | |
781 "Unknown file type for '%s'.\n"); | |
782 | |
783 GtkWidget *dialog; | |
784 | |
785 g_return_if_fail(GTK_IS_WINDOW(parent)); | |
786 g_return_if_fail(filename != NULL); | |
787 | |
788 dialog = | |
789 gtk_message_dialog_new_with_markup(GTK_WINDOW(parent), | |
790 GTK_DIALOG_DESTROY_WITH_PARENT, | |
791 GTK_MESSAGE_ERROR, | |
792 GTK_BUTTONS_OK, | |
793 _(markup), | |
794 filename); | |
795 gtk_dialog_run(GTK_DIALOG(dialog)); | |
796 gtk_widget_destroy(dialog); | |
797 } | |
798 | |
799 static void | |
800 playlistwin_save_playlist(const gchar * filename) | |
801 { | |
802 PlaylistContainer *plc; | |
803 gchar *ext = strrchr(filename, '.') + 1; | |
804 | |
805 plc = playlist_container_find(ext); | |
806 if (plc == NULL) { | |
807 show_playlist_save_format_error(GTK_WINDOW(playlistwin), filename); | |
808 return; | |
809 } | |
810 | |
811 str_replace_in(&cfg.playlist_path, g_path_get_dirname(filename)); | |
812 | |
813 if (g_file_test(filename, G_FILE_TEST_IS_REGULAR)) | |
814 if (!show_playlist_overwrite_prompt(GTK_WINDOW(playlistwin), filename)) | |
815 return; | |
816 | |
817 if (!playlist_save(playlist_get_active(), filename)) | |
818 show_playlist_save_error(GTK_WINDOW(playlistwin), filename); | |
819 } | |
820 | |
821 static void | |
822 playlistwin_load_playlist(const gchar * filename) | |
823 { | |
824 Playlist *playlist = playlist_get_active(); | |
825 | |
826 g_return_if_fail(filename != NULL); | |
827 | |
828 str_replace_in(&cfg.playlist_path, g_strdup(filename)); | |
829 | |
830 playlist_clear(playlist); | |
831 mainwin_clear_song_info(); | |
832 mainwin_set_info_text(); | |
833 | |
834 playlist_load(playlist, filename); | |
835 playlist_set_current_name(playlist, filename); | |
836 } | |
837 | |
838 static gchar * | |
839 playlist_file_selection_load(const gchar * title, | |
840 const gchar * default_filename) | |
841 { | |
842 static GtkWidget *dialog = NULL; | |
843 GtkWidget *button; | |
844 gchar *filename; | |
845 | |
846 g_return_val_if_fail(title != NULL, NULL); | |
847 | |
848 if(!dialog) { | |
849 dialog = gtk_file_chooser_dialog_new(title, GTK_WINDOW(mainwin), | |
850 GTK_FILE_CHOOSER_ACTION_OPEN, NULL, NULL); | |
851 | |
852 if (default_filename) | |
853 gtk_file_chooser_set_filename(GTK_FILE_CHOOSER(dialog), | |
854 default_filename); | |
855 | |
856 button = gtk_dialog_add_button(GTK_DIALOG(dialog), GTK_STOCK_CANCEL, | |
857 GTK_RESPONSE_REJECT); | |
858 gtk_button_set_use_stock(GTK_BUTTON(button), TRUE); | |
859 GTK_WIDGET_SET_FLAGS(button, GTK_CAN_DEFAULT); | |
860 | |
861 button = gtk_dialog_add_button(GTK_DIALOG(dialog), | |
862 GTK_STOCK_OPEN, | |
863 GTK_RESPONSE_ACCEPT); | |
864 gtk_button_set_use_stock(GTK_BUTTON(button), TRUE); | |
865 gtk_dialog_set_default_response(GTK_DIALOG(dialog), GTK_RESPONSE_ACCEPT); | |
866 } | |
867 | |
868 if (gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) | |
869 filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog)); | |
870 else | |
871 filename = NULL; | |
872 | |
873 gtk_widget_hide(dialog); | |
874 return filename; | |
875 } | |
876 | |
877 static gchar * | |
878 playlist_file_selection_save(const gchar * title, | |
879 const gchar * default_filename) | |
880 { | |
881 static GtkWidget *dialog = NULL; | |
882 GtkWidget *button; | |
883 gchar *filename; | |
884 | |
885 g_return_val_if_fail(title != NULL, NULL); | |
886 | |
887 if(!dialog) { | |
888 dialog = gtk_file_chooser_dialog_new(title, GTK_WINDOW(mainwin), | |
889 GTK_FILE_CHOOSER_ACTION_SAVE, NULL, NULL); | |
890 | |
891 if (default_filename) | |
892 gtk_file_chooser_set_filename(GTK_FILE_CHOOSER(dialog), | |
893 default_filename); | |
894 | |
895 button = gtk_dialog_add_button(GTK_DIALOG(dialog), GTK_STOCK_CANCEL, | |
896 GTK_RESPONSE_REJECT); | |
897 gtk_button_set_use_stock(GTK_BUTTON(button), TRUE); | |
898 GTK_WIDGET_SET_FLAGS(button, GTK_CAN_DEFAULT); | |
899 | |
900 button = gtk_dialog_add_button(GTK_DIALOG(dialog), | |
901 GTK_STOCK_SAVE, | |
902 GTK_RESPONSE_ACCEPT); | |
903 gtk_button_set_use_stock(GTK_BUTTON(button), TRUE); | |
904 gtk_dialog_set_default_response(GTK_DIALOG(dialog), GTK_RESPONSE_ACCEPT); | |
905 } | |
906 | |
907 if (gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) | |
908 filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog)); | |
909 else | |
910 filename = NULL; | |
911 | |
912 gtk_widget_hide(dialog); | |
913 return filename; | |
914 } | |
915 | |
916 void | |
917 playlistwin_select_playlist_to_load(const gchar * default_filename) | |
918 { | |
919 gchar *filename = | |
920 playlist_file_selection_load(_("Load Playlist"), default_filename); | |
921 | |
922 if (filename) { | |
923 playlistwin_load_playlist(filename); | |
924 g_free(filename); | |
925 } | |
926 } | |
927 | |
928 static void | |
929 playlistwin_select_playlist_to_save(const gchar * default_filename) | |
930 { | |
931 gchar *dot = NULL, *basename = NULL; | |
932 gchar *filename = | |
933 playlist_file_selection_save(_("Save Playlist"), default_filename); | |
934 | |
935 if (filename) { | |
936 /* Default to xspf if no filename has extension */ | |
937 basename = g_path_get_basename(filename); | |
938 dot = strrchr(basename, '.'); | |
939 if( dot == NULL || dot == basename) { | |
940 gchar *oldname = filename; | |
941 filename = g_strconcat(oldname, ".xspf", NULL); | |
942 g_free(oldname); | |
943 } | |
944 g_free(basename); | |
945 | |
946 playlistwin_save_playlist(filename); | |
947 g_free(filename); | |
948 } | |
949 } | |
950 | |
951 static gboolean | |
952 inside_sensitive_widgets(gint x, gint y) | |
953 { | |
954 return (widget_contains(WIDGET(playlistwin_list), x, y) || | |
955 widget_contains(WIDGET(playlistwin_slider), x, y) || | |
956 widget_contains(WIDGET(playlistwin_close), x, y) || | |
957 widget_contains(WIDGET(playlistwin_shade), x, y) || | |
958 widget_contains(WIDGET(playlistwin_time_min), x, y) || | |
959 widget_contains(WIDGET(playlistwin_time_sec), x, y) || | |
960 widget_contains(WIDGET(playlistwin_info), x, y) || | |
961 widget_contains(WIDGET(playlistwin_srew), x, y) || | |
962 widget_contains(WIDGET(playlistwin_splay), x, y) || | |
963 widget_contains(WIDGET(playlistwin_spause), x, y) || | |
964 widget_contains(WIDGET(playlistwin_sstop), x, y) || | |
965 widget_contains(WIDGET(playlistwin_sfwd), x, y) || | |
966 widget_contains(WIDGET(playlistwin_seject), x, y) || | |
967 widget_contains(WIDGET(playlistwin_sscroll_up), x, y) || | |
968 widget_contains(WIDGET(playlistwin_sscroll_down), x, y)); | |
969 } | |
970 | |
971 #define REGION_L(x1,x2,y1,y2) \ | |
972 (event->x >= (x1) && event->x < (x2) && \ | |
973 event->y >= cfg.playlist_height - (y1) && \ | |
974 event->y < cfg.playlist_height - (y2)) | |
975 | |
976 #define REGION_R(x1,x2,y1,y2) \ | |
977 (event->x >= playlistwin_get_width() - (x1) && \ | |
978 event->x < playlistwin_get_width() - (x2) && \ | |
979 event->y >= cfg.playlist_height - (y1) && \ | |
980 event->y < cfg.playlist_height - (y2)) | |
981 | |
982 static void | |
983 playlistwin_scrolled(GtkWidget * widget, | |
984 GdkEventScroll * event, | |
985 gpointer callback_data) | |
986 { | |
987 | |
988 if (event->direction == GDK_SCROLL_DOWN) | |
989 playlistwin_scroll(cfg.scroll_pl_by); | |
990 | |
991 if (event->direction == GDK_SCROLL_UP) | |
992 playlistwin_scroll(-cfg.scroll_pl_by); | |
993 | |
994 g_cond_signal(cond_scan); | |
995 | |
996 } | |
997 | |
998 static gboolean | |
999 playlistwin_press(GtkWidget * widget, | |
1000 GdkEventButton * event, | |
1001 gpointer callback_data) | |
1002 { | |
1003 gboolean grab = TRUE; | |
1004 gint xpos, ypos; | |
1005 GtkRequisition req; | |
1006 | |
1007 gtk_window_get_position(GTK_WINDOW(playlistwin), &xpos, &ypos); | |
1008 | |
1009 if (event->button == 1 && !cfg.show_wm_decorations && | |
1010 ((!cfg.playlist_shaded && | |
1011 event->x > playlistwin_get_width() - 20 && | |
1012 event->y > cfg.playlist_height - 20) || | |
1013 (cfg.playlist_shaded && | |
1014 event->x >= playlistwin_get_width() - 31 && | |
1015 event->x < playlistwin_get_width() - 22))) { | |
1016 | |
1017 /* NOTE: Workaround for bug #214 */ | |
1018 if (event->type != GDK_2BUTTON_PRESS && | |
1019 event->type != GDK_3BUTTON_PRESS) { | |
1020 /* resize area */ | |
1021 playlistwin_resizing = TRUE; | |
1022 gtk_window_begin_resize_drag(GTK_WINDOW(widget), | |
1023 GDK_WINDOW_EDGE_SOUTH_EAST, | |
1024 event->button, | |
1025 event->x + xpos, event->y + ypos, | |
1026 event->time); | |
1027 } | |
1028 grab = FALSE; | |
1029 } | |
1030 else if (event->button == 1 && REGION_L(12, 37, 29, 11)) { | |
1031 /* ADD button menu */ | |
1032 gtk_widget_size_request(playlistwin_pladd_menu, &req); | |
1033 ui_manager_popup_menu_show(GTK_MENU(playlistwin_pladd_menu), | |
1034 xpos + 12, | |
1035 (ypos + playlistwin_get_height()) - 8 - req.height, | |
1036 event->button, | |
1037 event->time); | |
1038 grab = FALSE; | |
1039 } | |
1040 else if (event->button == 1 && REGION_L(41, 66, 29, 11)) { | |
1041 /* SUB button menu */ | |
1042 gtk_widget_size_request(playlistwin_pldel_menu, &req); | |
1043 ui_manager_popup_menu_show(GTK_MENU(playlistwin_pldel_menu), | |
1044 xpos + 40, | |
1045 (ypos + playlistwin_get_height()) - 8 - req.height, | |
1046 event->button, | |
1047 event->time); | |
1048 grab = FALSE; | |
1049 } | |
1050 else if (event->button == 1 && REGION_L(70, 95, 29, 11)) { | |
1051 /* SEL button menu */ | |
1052 gtk_widget_size_request(playlistwin_plsel_menu, &req); | |
1053 ui_manager_popup_menu_show(GTK_MENU(playlistwin_plsel_menu), | |
1054 xpos + 68, | |
1055 (ypos + playlistwin_get_height()) - 8 - req.height, | |
1056 event->button, | |
1057 event->time); | |
1058 grab = FALSE; | |
1059 } | |
1060 else if (event->button == 1 && REGION_L(99, 124, 29, 11)) { | |
1061 /* MISC button menu */ | |
1062 gtk_widget_size_request(playlistwin_plsort_menu, &req); | |
1063 ui_manager_popup_menu_show(GTK_MENU(playlistwin_plsort_menu), | |
1064 xpos + 100, | |
1065 (ypos + playlistwin_get_height()) - 8 - req.height, | |
1066 event->button, | |
1067 event->time); | |
1068 grab = FALSE; | |
1069 } | |
1070 else if (event->button == 1 && REGION_R(46, 23, 29, 11)) { | |
1071 /* LIST button menu */ | |
1072 gtk_widget_size_request(playlistwin_pllist_menu, &req); | |
1073 ui_manager_popup_menu_show(GTK_MENU(playlistwin_pllist_menu), | |
1074 xpos + playlistwin_get_width() - req.width - 12, | |
1075 (ypos + playlistwin_get_height()) - 8 - req.height, | |
1076 event->button, | |
1077 event->time); | |
1078 grab = FALSE; | |
1079 } | |
1080 else if (event->button == 1 && REGION_R(82, 54, 15, 9)) { | |
1081 if (cfg.timer_mode == TIMER_ELAPSED) | |
1082 cfg.timer_mode = TIMER_REMAINING; | |
1083 else | |
1084 cfg.timer_mode = TIMER_ELAPSED; | |
1085 } | |
1086 else if (event->button == 2 && (event->type == GDK_BUTTON_PRESS) && | |
1087 widget_contains(WIDGET(playlistwin_list), event->x, event->y)) { | |
1088 gtk_selection_convert(widget, GDK_SELECTION_PRIMARY, | |
1089 GDK_TARGET_STRING, event->time); | |
1090 } | |
1091 else if (event->button == 1 && event->type == GDK_BUTTON_PRESS && | |
1092 !inside_sensitive_widgets(event->x, event->y) && (cfg.easy_move || event->y < 14)) | |
1093 { | |
1094 dock_move_press(dock_window_list, GTK_WINDOW(playlistwin), event, | |
1095 FALSE); | |
1096 gtk_window_present(GTK_WINDOW(playlistwin)); | |
1097 } | |
1098 else if (event->button == 1 && event->type == GDK_2BUTTON_PRESS && | |
1099 !inside_sensitive_widgets(event->x, event->y) | |
1100 && event->y < 14) { | |
1101 /* double click on title bar */ | |
1102 playlistwin_shade_toggle(); | |
1103 if (dock_is_moving(GTK_WINDOW(playlistwin))) | |
1104 dock_move_release(GTK_WINDOW(playlistwin)); | |
1105 return TRUE; | |
1106 } | |
1107 else if (event->button == 3 && | |
1108 !(widget_contains(WIDGET(playlistwin_list), event->x, event->y) || | |
1109 (event->y >= cfg.playlist_height - 29 && | |
1110 event->y < cfg.playlist_height - 11 && | |
1111 ((event->x >= 12 && event->x < 37) || | |
1112 (event->x >= 41 && event->x < 66) || | |
1113 (event->x >= 70 && event->x < 95) || | |
1114 (event->x >= 99 && event->x < 124) || | |
1115 (event->x >= playlistwin_get_width() - 46 && | |
1116 event->x < playlistwin_get_width() - 23))))) { | |
1117 /* | |
1118 * Pop up the main menu a few pixels down to avoid | |
1119 * anything to be selected initially. | |
1120 */ | |
1121 ui_manager_popup_menu_show(GTK_MENU(mainwin_general_menu), event->x_root, | |
1122 event->y_root + 2, 3, event->time); | |
1123 grab = FALSE; | |
1124 } | |
1125 else if (event->button == 3 && | |
1126 widget_contains(WIDGET(playlistwin_list), event->x, event->y)) { | |
2356
4b2c7d9523e7
[svn] right-clicking on playlist entry automatically selects the entry if it was unselected before, closes #728
mf0102
parents:
2351
diff
changeset
|
1127 handle_press_cb(playlistwin_wlist, widget, event); |
2313 | 1128 ui_manager_popup_menu_show(GTK_MENU(playlistwin_popup_menu), |
1129 event->x_root, event->y_root + 5, | |
1130 event->button, event->time); | |
1131 grab = FALSE; | |
1132 } | |
1133 else { | |
1134 handle_press_cb(playlistwin_wlist, widget, event); | |
1135 draw_playlist_window(FALSE); | |
1136 } | |
1137 | |
1138 if (grab) | |
1139 gdk_pointer_grab(playlistwin->window, FALSE, | |
1140 GDK_BUTTON_MOTION_MASK | GDK_BUTTON_RELEASE_MASK | | |
1141 GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK | | |
1142 GDK_BUTTON1_MOTION_MASK, NULL, NULL, | |
1143 GDK_CURRENT_TIME); | |
1144 | |
1145 return FALSE; | |
1146 } | |
1147 | |
1148 static gboolean | |
1149 playlistwin_focus_in(GtkWidget * widget, GdkEvent * event, gpointer data) | |
1150 { | |
1151 playlistwin_close->pb_allow_draw = TRUE; | |
1152 playlistwin_shade->pb_allow_draw = TRUE; | |
1153 draw_playlist_window(TRUE); | |
1154 return FALSE; | |
1155 } | |
1156 | |
1157 static gboolean | |
1158 playlistwin_focus_out(GtkWidget * widget, | |
1159 GdkEventButton * event, gpointer data) | |
1160 { | |
1161 playlistwin_close->pb_allow_draw = FALSE; | |
1162 playlistwin_shade->pb_allow_draw = FALSE; | |
1163 draw_playlist_window(TRUE); | |
1164 return FALSE; | |
1165 } | |
1166 | |
1167 static gboolean | |
1168 playlistwin_configure(GtkWidget * window, | |
1169 GdkEventConfigure * event, gpointer data) | |
1170 { | |
1171 if (!GTK_WIDGET_VISIBLE(window)) | |
1172 return FALSE; | |
1173 | |
1174 cfg.playlist_x = event->x; | |
1175 cfg.playlist_y = event->y; | |
1176 | |
1177 if (playlistwin_resizing) { | |
1178 if (event->width != playlistwin_get_width() || | |
1179 event->height != playlistwin_get_height()) | |
1180 playlistwin_resize(event->width, event->height); | |
1181 } | |
1182 return TRUE; | |
1183 } | |
1184 | |
1185 void | |
1186 playlistwin_set_back_pixmap(void) | |
1187 { | |
1188 gdk_window_set_back_pixmap(playlistwin->window, playlistwin_bg, 0); | |
1189 gdk_window_clear(playlistwin->window); | |
1190 } | |
1191 | |
1192 static gboolean | |
1193 playlistwin_delete(GtkWidget * w, gpointer data) | |
1194 { | |
1195 playlistwin_hide(); | |
1196 return TRUE; | |
1197 } | |
1198 | |
1199 static void | |
1200 playlistwin_keypress_up_down_handler(PlayList_List * pl, | |
1201 gboolean up, guint state) | |
1202 { | |
1203 Playlist *playlist = playlist_get_active(); | |
1204 | |
1205 if ((state & GDK_MOD1_MASK) && (state & GDK_SHIFT_MASK)) | |
1206 return; | |
1207 if (!(state & GDK_MOD1_MASK)) | |
1208 playlist_select_all(playlist, FALSE); | |
1209 | |
1210 if (pl->pl_prev_selected == -1 || | |
1211 (!playlistwin_item_visible(pl->pl_prev_selected) && | |
1212 !(state & GDK_SHIFT_MASK && pl->pl_prev_min != -1))) { | |
1213 pl->pl_prev_selected = pl->pl_first; | |
1214 } | |
1215 else if (state & GDK_SHIFT_MASK) { | |
1216 if (pl->pl_prev_min == -1) { | |
1217 pl->pl_prev_max = pl->pl_prev_selected; | |
1218 pl->pl_prev_min = pl->pl_prev_selected; | |
1219 } | |
1220 pl->pl_prev_max += (up ? -1 : 1); | |
1221 pl->pl_prev_max = | |
1222 CLAMP(pl->pl_prev_max, 0, playlist_get_length(playlist) - 1); | |
1223 | |
1224 pl->pl_first = MIN(pl->pl_first, pl->pl_prev_max); | |
1225 pl->pl_first = MAX(pl->pl_first, pl->pl_prev_max - | |
1226 pl->pl_num_visible + 1); | |
1227 playlist_select_range(playlist, pl->pl_prev_min, pl->pl_prev_max, TRUE); | |
1228 return; | |
1229 } | |
1230 else if (state & GDK_MOD1_MASK) { | |
1231 if (up) | |
1232 playlist_list_move_up(pl); | |
1233 else | |
1234 playlist_list_move_down(pl); | |
1235 if (pl->pl_prev_min < pl->pl_first) | |
1236 pl->pl_first = pl->pl_prev_min; | |
1237 else if (pl->pl_prev_max >= (pl->pl_first + pl->pl_num_visible)) | |
1238 pl->pl_first = pl->pl_prev_max - pl->pl_num_visible + 1; | |
1239 return; | |
1240 } | |
1241 else if (up) | |
1242 pl->pl_prev_selected--; | |
1243 else | |
1244 pl->pl_prev_selected++; | |
1245 | |
1246 pl->pl_prev_selected = | |
1247 CLAMP(pl->pl_prev_selected, 0, playlist_get_length(playlist) - 1); | |
1248 | |
1249 if (pl->pl_prev_selected < pl->pl_first) | |
1250 pl->pl_first--; | |
1251 else if (pl->pl_prev_selected >= (pl->pl_first + pl->pl_num_visible)) | |
1252 pl->pl_first++; | |
1253 | |
1254 playlist_select_range(playlist, pl->pl_prev_selected, pl->pl_prev_selected, TRUE); | |
1255 pl->pl_prev_min = -1; | |
1256 } | |
1257 | |
1258 /* FIXME: Handle the keys through menu */ | |
1259 | |
1260 static gboolean | |
1261 playlistwin_keypress(GtkWidget * w, GdkEventKey * event, gpointer data) | |
1262 { | |
1263 Playlist *playlist = playlist_get_active(); | |
1264 | |
1265 guint keyval; | |
1266 gboolean refresh = FALSE; | |
1267 | |
1268 if (cfg.playlist_shaded) | |
1269 return FALSE; | |
1270 | |
1271 switch (keyval = event->keyval) { | |
1272 case GDK_KP_Up: | |
1273 case GDK_KP_Down: | |
1274 case GDK_Up: | |
1275 case GDK_Down: | |
1276 playlistwin_keypress_up_down_handler(playlistwin_list, | |
1277 keyval == GDK_Up | |
1278 || keyval == GDK_KP_Up, | |
1279 event->state); | |
1280 refresh = TRUE; | |
1281 break; | |
1282 case GDK_Page_Up: | |
1283 playlistwin_scroll(-playlistwin_list->pl_num_visible); | |
1284 refresh = TRUE; | |
1285 break; | |
1286 case GDK_Page_Down: | |
1287 playlistwin_scroll(playlistwin_list->pl_num_visible); | |
1288 refresh = TRUE; | |
1289 break; | |
1290 case GDK_Home: | |
1291 playlistwin_list->pl_first = 0; | |
1292 refresh = TRUE; | |
1293 break; | |
1294 case GDK_End: | |
1295 playlistwin_list->pl_first = | |
1296 playlist_get_length(playlist) - playlistwin_list->pl_num_visible; | |
1297 refresh = TRUE; | |
1298 break; | |
1299 case GDK_Return: | |
1300 if (playlistwin_list->pl_prev_selected > -1 | |
1301 && playlistwin_item_visible(playlistwin_list->pl_prev_selected)) { | |
1302 playlist_set_position(playlist, playlistwin_list->pl_prev_selected); | |
1303 if (!playback_get_playing()) | |
1304 playback_initiate(); | |
1305 } | |
1306 break; | |
1307 case GDK_3: | |
1308 if (event->state & GDK_CONTROL_MASK) | |
1309 playlistwin_fileinfo(); | |
1310 break; | |
1311 case GDK_Delete: | |
1312 if (event->state & GDK_CONTROL_MASK) | |
1313 playlist_delete(playlist, TRUE); | |
1314 else | |
1315 playlist_delete(playlist, FALSE); | |
1316 break; | |
1317 case GDK_Insert: | |
1318 if (event->state & GDK_MOD1_MASK) | |
1319 mainwin_show_add_url_window(); | |
1320 else | |
1321 playlistwin_show_filebrowser(); | |
1322 break; | |
1323 case GDK_Left: | |
1324 case GDK_KP_Left: | |
1325 case GDK_KP_7: | |
1326 if (playlist_get_current_length(playlist) != -1) | |
1327 playback_seek(CLAMP | |
1328 (playback_get_time() - 5000, 0, | |
1329 playlist_get_current_length(playlist)) / 1000); | |
1330 break; | |
1331 case GDK_Right: | |
1332 case GDK_KP_Right: | |
1333 case GDK_KP_9: | |
1334 if (playlist_get_current_length(playlist) != -1) | |
1335 playback_seek(CLAMP | |
1336 (playback_get_time() + 5000, 0, | |
1337 playlist_get_current_length(playlist)) / 1000); | |
1338 break; | |
1339 case GDK_KP_4: | |
1340 playlist_prev(playlist); | |
1341 break; | |
1342 case GDK_KP_6: | |
1343 playlist_next(playlist); | |
1344 break; | |
1345 | |
1346 case GDK_Escape: | |
1347 mainwin_minimize_cb(); | |
1348 break; | |
1349 default: | |
1350 return FALSE; | |
1351 } | |
1352 | |
1353 if (refresh) { | |
1354 g_cond_signal(cond_scan); | |
1355 playlistwin_update_list(playlist_get_active()); | |
1356 } | |
1357 | |
1358 return TRUE; | |
1359 } | |
1360 | |
1361 static void | |
1362 playlistwin_draw_frame(void) | |
1363 { | |
1364 gboolean focus = | |
1365 gtk_window_has_toplevel_focus(GTK_WINDOW(playlistwin)) || | |
1366 !cfg.dim_titlebar; | |
1367 | |
1368 if (cfg.playlist_shaded) { | |
1369 skin_draw_playlistwin_shaded(bmp_active_skin, | |
1370 playlistwin_bg, playlistwin_gc, | |
1371 playlistwin_get_width(), focus); | |
1372 } | |
1373 else { | |
1374 skin_draw_playlistwin_frame(bmp_active_skin, | |
1375 playlistwin_bg, playlistwin_gc, | |
1376 playlistwin_get_width(), | |
1377 cfg.playlist_height, focus); | |
1378 } | |
1379 } | |
1380 | |
1381 void | |
1382 draw_playlist_window(gboolean force) | |
1383 { | |
1384 gboolean redraw; | |
1385 GList *wl; | |
1386 Widget *w; | |
1387 | |
1388 if (force) | |
1389 playlistwin_draw_frame(); | |
1390 | |
1391 widget_list_lock(playlistwin_wlist); | |
1392 widget_list_draw(playlistwin_wlist, &redraw, force); | |
1393 | |
1394 if (redraw || force) { | |
1395 if (force) { | |
1396 gdk_window_clear(playlistwin->window); | |
1397 } | |
1398 else { | |
1399 for (wl = playlistwin_wlist; wl; wl = g_list_next(wl)) { | |
1400 w = WIDGET(wl->data); | |
1401 if (w->redraw && w->visible) { | |
1402 gdk_window_clear_area(playlistwin->window, w->x, w->y, | |
1403 w->width, w->height); | |
1404 w->redraw = FALSE; | |
1405 } | |
1406 } | |
1407 } | |
1408 | |
1409 gdk_flush(); | |
1410 } | |
1411 | |
1412 widget_list_unlock(playlistwin_wlist); | |
1413 } | |
1414 | |
1415 | |
1416 void | |
1417 playlistwin_hide_timer(void) | |
1418 { | |
1419 textbox_set_text(playlistwin_time_min, " "); | |
1420 textbox_set_text(playlistwin_time_sec, " "); | |
1421 } | |
1422 | |
1423 void | |
1424 playlistwin_set_time(gint time, gint length, TimerMode mode) | |
1425 { | |
1426 gchar *text, sign; | |
1427 | |
1428 if (mode == TIMER_REMAINING && length != -1) { | |
1429 time = length - time; | |
1430 sign = '-'; | |
1431 } | |
1432 else | |
1433 sign = ' '; | |
1434 | |
1435 time /= 1000; | |
1436 | |
1437 if (time < 0) | |
1438 time = 0; | |
1439 if (time > 99 * 60) | |
1440 time /= 60; | |
1441 | |
1442 text = g_strdup_printf("%c%-2.2d", sign, time / 60); | |
1443 textbox_set_text(playlistwin_time_min, text); | |
1444 g_free(text); | |
1445 | |
1446 text = g_strdup_printf("%-2.2d", time % 60); | |
1447 textbox_set_text(playlistwin_time_sec, text); | |
1448 g_free(text); | |
1449 } | |
1450 | |
1451 static void | |
1452 playlistwin_drag_motion(GtkWidget * widget, | |
1453 GdkDragContext * context, | |
1454 gint x, gint y, | |
1455 GtkSelectionData * selection_data, | |
1456 guint info, guint time, gpointer user_data) | |
1457 { | |
1458 playlistwin_list->pl_drag_motion = TRUE; | |
1459 playlistwin_list->drag_motion_x = x; | |
1460 playlistwin_list->drag_motion_y = y; | |
1461 playlistwin_update_list(playlist_get_active()); | |
1462 playlistwin_hint_flag = TRUE; | |
1463 } | |
1464 | |
1465 static void | |
1466 playlistwin_drag_end(GtkWidget * widget, | |
1467 GdkDragContext * context, gpointer user_data) | |
1468 { | |
1469 playlistwin_list->pl_drag_motion = FALSE; | |
1470 playlistwin_hint_flag = FALSE; | |
1471 playlistwin_update_list(playlist_get_active()); | |
1472 } | |
1473 | |
1474 static void | |
1475 playlistwin_drag_data_received(GtkWidget * widget, | |
1476 GdkDragContext * context, | |
1477 gint x, gint y, | |
1478 GtkSelectionData * | |
1479 selection_data, guint info, | |
1480 guint time, gpointer user_data) | |
1481 { | |
1482 gint pos; | |
1483 Playlist *playlist = playlist_get_active(); | |
1484 | |
2363 | 1485 g_return_if_fail(selection_data); |
2313 | 1486 |
1487 if (!selection_data->data) { | |
1488 g_message("Received no DND data!"); | |
1489 return; | |
1490 } | |
1491 | |
1492 if (widget_contains(WIDGET(playlistwin_list), x, y)) { | |
1493 pos = (y - WIDGET(playlistwin_list)->y) / | |
1494 playlistwin_list->pl_fheight + playlistwin_list->pl_first; | |
1495 | |
1496 pos = MIN(pos, playlist_get_length(playlist)); | |
1497 playlist_ins_url(playlist, (gchar *) selection_data->data, pos); | |
1498 } | |
1499 else | |
1500 playlist_add_url(playlist, (gchar *) selection_data->data); | |
1501 } | |
1502 | |
1503 static void | |
1504 local_playlist_prev(void) | |
1505 { | |
1506 playlist_prev(playlist_get_active()); | |
1507 } | |
1508 | |
1509 static void | |
1510 local_playlist_next(void) | |
1511 { | |
1512 playlist_next(playlist_get_active()); | |
1513 } | |
1514 | |
1515 static void | |
1516 playlistwin_create_widgets(void) | |
1517 { | |
1518 /* This function creates the custom widgets used by the playlist editor */ | |
1519 | |
1520 /* text box for displaying song title in shaded mode */ | |
1521 playlistwin_sinfo = | |
1522 create_textbox(&playlistwin_wlist, playlistwin_bg, playlistwin_gc, | |
1523 4, 4, playlistwin_get_width() - 35, TRUE, SKIN_TEXT); | |
1524 | |
1525 playlistwin_set_sinfo_font(cfg.playlist_font); | |
1526 | |
1527 if (!cfg.playlist_shaded) | |
1528 widget_hide(WIDGET(playlistwin_sinfo)); | |
1529 | |
1530 /* shade/unshade window push button */ | |
1531 if (cfg.playlist_shaded) | |
1532 playlistwin_shade = | |
1533 create_pbutton(&playlistwin_wlist, playlistwin_bg, | |
1534 playlistwin_gc, playlistwin_get_width() - 21, 3, | |
1535 9, 9, 128, 45, 150, 42, | |
1536 playlistwin_shade_toggle, SKIN_PLEDIT); | |
1537 else | |
1538 playlistwin_shade = | |
1539 create_pbutton(&playlistwin_wlist, playlistwin_bg, | |
1540 playlistwin_gc, playlistwin_get_width() - 21, 3, | |
1541 9, 9, 157, 3, 62, 42, playlistwin_shade_toggle, | |
1542 SKIN_PLEDIT); | |
1543 | |
1544 playlistwin_shade->pb_allow_draw = FALSE; | |
1545 | |
1546 /* close window push button */ | |
1547 playlistwin_close = | |
1548 create_pbutton(&playlistwin_wlist, playlistwin_bg, playlistwin_gc, | |
1549 playlistwin_get_width() - 11, 3, 9, 9, | |
1550 cfg.playlist_shaded ? 138 : 167, | |
1551 cfg.playlist_shaded ? 45 : 3, 52, 42, | |
1552 playlistwin_hide, SKIN_PLEDIT); | |
1553 playlistwin_close->pb_allow_draw = FALSE; | |
1554 | |
1555 /* playlist list box */ | |
1556 playlistwin_list = | |
1557 create_playlist_list(&playlistwin_wlist, playlistwin_bg, | |
1558 playlistwin_gc, 12, 20, | |
1559 playlistwin_get_width() - 31, | |
1560 cfg.playlist_height - 58); | |
1561 playlist_list_set_font(cfg.playlist_font); | |
1562 | |
1563 /* playlist list box slider */ | |
1564 playlistwin_slider = | |
1565 create_playlistslider(&playlistwin_wlist, playlistwin_bg, | |
1566 playlistwin_gc, playlistwin_get_width() - 15, | |
1567 20, cfg.playlist_height - 58, playlistwin_list); | |
1568 /* track time (minute) */ | |
1569 playlistwin_time_min = | |
1570 create_textbox(&playlistwin_wlist, playlistwin_bg, playlistwin_gc, | |
1571 playlistwin_get_width() - 82, | |
1572 cfg.playlist_height - 15, 15, FALSE, SKIN_TEXT); | |
1573 | |
1574 /* track time (second) */ | |
1575 playlistwin_time_sec = | |
1576 create_textbox(&playlistwin_wlist, playlistwin_bg, playlistwin_gc, | |
1577 playlistwin_get_width() - 64, | |
1578 cfg.playlist_height - 15, 10, FALSE, SKIN_TEXT); | |
1579 | |
1580 /* playlist information (current track length / total track length) */ | |
1581 playlistwin_info = | |
1582 create_textbox(&playlistwin_wlist, playlistwin_bg, playlistwin_gc, | |
1583 playlistwin_get_width() - 143, | |
1584 cfg.playlist_height - 28, 90, FALSE, SKIN_TEXT); | |
1585 | |
1586 /* mini play control buttons at right bottom corner */ | |
1587 | |
1588 /* rewind button */ | |
1589 playlistwin_srew = | |
1590 create_sbutton(&playlistwin_wlist, playlistwin_bg, playlistwin_gc, | |
1591 playlistwin_get_width() - 144, | |
1592 cfg.playlist_height - 16, 8, 7, local_playlist_prev); | |
1593 | |
1594 /* play button */ | |
1595 playlistwin_splay = | |
1596 create_sbutton(&playlistwin_wlist, playlistwin_bg, playlistwin_gc, | |
1597 playlistwin_get_width() - 138, | |
1598 cfg.playlist_height - 16, 10, 7, mainwin_play_pushed); | |
1599 | |
1600 /* pause button */ | |
1601 playlistwin_spause = | |
1602 create_sbutton(&playlistwin_wlist, playlistwin_bg, playlistwin_gc, | |
1603 playlistwin_get_width() - 128, | |
1604 cfg.playlist_height - 16, 10, 7, playback_pause); | |
1605 | |
1606 /* stop button */ | |
1607 playlistwin_sstop = | |
1608 create_sbutton(&playlistwin_wlist, playlistwin_bg, playlistwin_gc, | |
1609 playlistwin_get_width() - 118, | |
1610 cfg.playlist_height - 16, 9, 7, mainwin_stop_pushed); | |
1611 | |
1612 /* forward button */ | |
1613 playlistwin_sfwd = | |
1614 create_sbutton(&playlistwin_wlist, playlistwin_bg, playlistwin_gc, | |
1615 playlistwin_get_width() - 109, | |
1616 cfg.playlist_height - 16, 8, 7, local_playlist_next); | |
1617 | |
1618 /* eject button */ | |
1619 playlistwin_seject = | |
1620 create_sbutton(&playlistwin_wlist, playlistwin_bg, playlistwin_gc, | |
1621 playlistwin_get_width() - 100, | |
1622 cfg.playlist_height - 16, 9, 7, mainwin_eject_pushed); | |
1623 | |
1624 | |
1625 playlistwin_sscroll_up = | |
1626 create_sbutton(&playlistwin_wlist, playlistwin_bg, playlistwin_gc, | |
1627 playlistwin_get_width() - 14, | |
1628 cfg.playlist_height - 35, 8, 5, | |
1629 playlistwin_scroll_up_pushed); | |
1630 playlistwin_sscroll_down = | |
1631 create_sbutton(&playlistwin_wlist, playlistwin_bg, playlistwin_gc, | |
1632 playlistwin_get_width() - 14, | |
1633 cfg.playlist_height - 30, 8, 5, | |
1634 playlistwin_scroll_down_pushed); | |
1635 | |
1636 } | |
1637 | |
1638 static void | |
1639 selection_received(GtkWidget * widget, | |
1640 GtkSelectionData * selection_data, gpointer data) | |
1641 { | |
1642 if (selection_data->type == GDK_SELECTION_TYPE_STRING && | |
1643 selection_data->length > 0) | |
1644 playlist_add_url(playlist_get_active(), (gchar *) selection_data->data); | |
1645 } | |
1646 | |
1647 static void | |
1648 playlistwin_create_window(void) | |
1649 { | |
1650 GdkPixbuf *icon; | |
1651 | |
1652 playlistwin = gtk_window_new(GTK_WINDOW_TOPLEVEL); | |
1653 gtk_window_set_title(GTK_WINDOW(playlistwin), _("Audacious Playlist Editor")); | |
1654 gtk_window_set_wmclass(GTK_WINDOW(playlistwin), "playlist", "Audacious"); | |
1655 gtk_window_set_role(GTK_WINDOW(playlistwin), "playlist"); | |
1656 gtk_window_set_default_size(GTK_WINDOW(playlistwin), | |
1657 playlistwin_get_width(), | |
1658 playlistwin_get_height()); | |
1659 gtk_window_set_resizable(GTK_WINDOW(playlistwin), TRUE); | |
1660 playlistwin_set_geometry_hints(cfg.playlist_shaded); | |
1661 dock_window_list = dock_window_set_decorated(dock_window_list, | |
1662 GTK_WINDOW(playlistwin), | |
1663 cfg.show_wm_decorations); | |
1664 | |
1665 gtk_window_set_transient_for(GTK_WINDOW(playlistwin), | |
1666 GTK_WINDOW(mainwin)); | |
1667 gtk_window_set_skip_taskbar_hint(GTK_WINDOW(playlistwin), TRUE); | |
1668 | |
1669 icon = gdk_pixbuf_new_from_xpm_data((const gchar **) bmp_playlist_icon); | |
1670 gtk_window_set_icon(GTK_WINDOW(playlistwin), icon); | |
1671 g_object_unref(icon); | |
1672 | |
1673 gtk_widget_set_app_paintable(playlistwin, TRUE); | |
1674 | |
1675 if (cfg.playlist_x != -1 && cfg.save_window_position) | |
1676 gtk_window_move(GTK_WINDOW(playlistwin), | |
1677 cfg.playlist_x, cfg.playlist_y); | |
1678 | |
1679 gtk_widget_add_events(playlistwin, GDK_POINTER_MOTION_MASK | | |
1680 GDK_FOCUS_CHANGE_MASK | GDK_BUTTON_MOTION_MASK | | |
1681 GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK | | |
1682 GDK_SCROLL_MASK | GDK_VISIBILITY_NOTIFY_MASK); | |
1683 gtk_widget_realize(playlistwin); | |
1684 | |
1685 util_set_cursor(playlistwin); | |
1686 | |
1687 g_signal_connect(playlistwin, "delete_event", | |
1688 G_CALLBACK(playlistwin_delete), NULL); | |
1689 g_signal_connect(playlistwin, "button_press_event", | |
1690 G_CALLBACK(playlistwin_press), NULL); | |
1691 g_signal_connect(playlistwin, "button_release_event", | |
1692 G_CALLBACK(playlistwin_release), NULL); | |
1693 g_signal_connect(playlistwin, "scroll_event", | |
1694 G_CALLBACK(playlistwin_scrolled), NULL); | |
1695 g_signal_connect(playlistwin, "motion_notify_event", | |
1696 G_CALLBACK(playlistwin_motion), NULL); | |
1697 g_signal_connect(playlistwin, "enter_notify_event", | |
1698 G_CALLBACK(playlistwin_enter), NULL); | |
1699 g_signal_connect(playlistwin, "leave_notify_event", | |
1700 G_CALLBACK(playlistwin_leave), NULL); | |
1701 g_signal_connect_after(playlistwin, "focus_in_event", | |
1702 G_CALLBACK(playlistwin_focus_in), NULL); | |
1703 g_signal_connect_after(playlistwin, "focus_out_event", | |
1704 G_CALLBACK(playlistwin_focus_out), NULL); | |
1705 g_signal_connect(playlistwin, "configure_event", | |
1706 G_CALLBACK(playlistwin_configure), NULL); | |
1707 g_signal_connect(playlistwin, "style_set", | |
1708 G_CALLBACK(playlistwin_set_back_pixmap), NULL); | |
1709 | |
1710 bmp_drag_dest_set(playlistwin); | |
1711 | |
1712 /* DnD stuff */ | |
1713 g_signal_connect(playlistwin, "drag-leave", | |
1714 G_CALLBACK(playlistwin_drag_end), NULL); | |
1715 g_signal_connect(playlistwin, "drag-data-delete", | |
1716 G_CALLBACK(playlistwin_drag_end), NULL); | |
1717 g_signal_connect(playlistwin, "drag-end", | |
1718 G_CALLBACK(playlistwin_drag_end), NULL); | |
1719 g_signal_connect(playlistwin, "drag-drop", | |
1720 G_CALLBACK(playlistwin_drag_end), NULL); | |
1721 g_signal_connect(playlistwin, "drag-data-received", | |
1722 G_CALLBACK(playlistwin_drag_data_received), NULL); | |
1723 g_signal_connect(playlistwin, "drag-motion", | |
1724 G_CALLBACK(playlistwin_drag_motion), NULL); | |
1725 | |
1726 g_signal_connect(playlistwin, "key_press_event", | |
1727 G_CALLBACK(playlistwin_keypress), NULL); | |
1728 g_signal_connect(playlistwin, "selection_received", | |
1729 G_CALLBACK(selection_received), NULL); | |
1730 | |
1731 playlistwin_set_mask(); | |
1732 } | |
1733 | |
1734 void | |
1735 playlistwin_create(void) | |
1736 { | |
1737 playlistwin_create_window(); | |
1738 | |
1739 /* create GC and back pixmap for custom widget to draw on */ | |
1740 playlistwin_gc = gdk_gc_new(playlistwin->window); | |
1741 playlistwin_bg = gdk_pixmap_new(playlistwin->window, | |
1742 playlistwin_get_width(), | |
1743 playlistwin_get_height_unshaded(), -1); | |
1744 gdk_window_set_back_pixmap(playlistwin->window, playlistwin_bg, 0); | |
1745 | |
1746 playlistwin_create_widgets(); | |
1747 playlistwin_update_info(playlist_get_active()); | |
1748 | |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
1749 playlistwin_infopopup = audacious_fileinfopopup_create(); |
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
1750 |
2313 | 1751 gtk_window_add_accel_group(GTK_WINDOW(playlistwin), ui_manager_get_accel_group()); |
1752 } | |
1753 | |
1754 | |
1755 void | |
1756 playlistwin_show(void) | |
1757 { | |
1758 GtkAction *action = gtk_action_group_get_action( | |
1759 toggleaction_group_others , "show playlist editor" ); | |
1760 gtk_toggle_action_set_active( GTK_TOGGLE_ACTION(action) , TRUE ); | |
1761 | |
1762 tbutton_set_toggled(mainwin_pl, TRUE); | |
1763 cfg.playlist_visible = TRUE; | |
1764 | |
1765 playlistwin_set_toprow(0); | |
1766 playlist_check_pos_current(playlist_get_active()); | |
1767 | |
2351
911743d27aba
[svn] - simplify and optimize the metadata tooltip trigger function
giacomo
parents:
2348
diff
changeset
|
1768 if ( playlistwin_infopopup_sid == 0 ) |
911743d27aba
[svn] - simplify and optimize the metadata tooltip trigger function
giacomo
parents:
2348
diff
changeset
|
1769 playlistwin_infopopup_sid = g_timeout_add( |
911743d27aba
[svn] - simplify and optimize the metadata tooltip trigger function
giacomo
parents:
2348
diff
changeset
|
1770 50 , (GSourceFunc)playlistwin_fileinfopopup_probe , playlistwin_infopopup ); |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
1771 |
2313 | 1772 gtk_widget_show(playlistwin); |
1773 } | |
1774 | |
1775 void | |
1776 playlistwin_hide(void) | |
1777 { | |
1778 GtkAction *action = gtk_action_group_get_action( | |
1779 toggleaction_group_others , "show playlist editor" ); | |
1780 gtk_toggle_action_set_active( GTK_TOGGLE_ACTION(action) , FALSE ); | |
1781 | |
1782 gtk_widget_hide(playlistwin); | |
1783 tbutton_set_toggled(mainwin_pl, FALSE); | |
1784 cfg.playlist_visible = FALSE; | |
1785 | |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
1786 /* no point in probing for playlistwin_infopopup trigger when the playlistwin is hidden */ |
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
1787 if ( playlistwin_infopopup_sid != 0 ) |
2351
911743d27aba
[svn] - simplify and optimize the metadata tooltip trigger function
giacomo
parents:
2348
diff
changeset
|
1788 { |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
1789 g_source_remove( playlistwin_infopopup_sid ); |
2351
911743d27aba
[svn] - simplify and optimize the metadata tooltip trigger function
giacomo
parents:
2348
diff
changeset
|
1790 playlistwin_infopopup_sid = 0; |
911743d27aba
[svn] - simplify and optimize the metadata tooltip trigger function
giacomo
parents:
2348
diff
changeset
|
1791 } |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
1792 |
2338
d0a04f1ee732
[svn] - in playlistwin_hide(), pass focus to the player main window only if it's visible
giacomo
parents:
2328
diff
changeset
|
1793 if ( cfg.player_visible ) |
d0a04f1ee732
[svn] - in playlistwin_hide(), pass focus to the player main window only if it's visible
giacomo
parents:
2328
diff
changeset
|
1794 { |
d0a04f1ee732
[svn] - in playlistwin_hide(), pass focus to the player main window only if it's visible
giacomo
parents:
2328
diff
changeset
|
1795 gtk_window_present(GTK_WINDOW(mainwin)); |
d0a04f1ee732
[svn] - in playlistwin_hide(), pass focus to the player main window only if it's visible
giacomo
parents:
2328
diff
changeset
|
1796 gtk_widget_grab_focus(mainwin); |
d0a04f1ee732
[svn] - in playlistwin_hide(), pass focus to the player main window only if it's visible
giacomo
parents:
2328
diff
changeset
|
1797 } |
2313 | 1798 } |
1799 | |
1800 void action_playlist_track_info(void) | |
1801 { | |
1802 playlistwin_fileinfo(); | |
1803 } | |
1804 | |
1805 void action_queue_toggle(void) | |
1806 { | |
1807 playlist_queue(playlist_get_active()); | |
1808 } | |
1809 | |
1810 void action_playlist_sort_by_playlist_entry(void) | |
1811 { | |
1812 Playlist *playlist = playlist_get_active(); | |
1813 | |
1814 playlist_sort(playlist, PLAYLIST_SORT_PLAYLIST); | |
1815 playlistwin_update_list(playlist); | |
1816 } | |
1817 | |
1818 void action_playlist_sort_by_track_number(void) | |
1819 { | |
1820 Playlist *playlist = playlist_get_active(); | |
1821 | |
1822 playlist_sort(playlist, PLAYLIST_SORT_TRACK); | |
1823 playlistwin_update_list(playlist); | |
1824 } | |
1825 | |
1826 void action_playlist_sort_by_title(void) | |
1827 { | |
1828 Playlist *playlist = playlist_get_active(); | |
1829 | |
1830 playlist_sort(playlist, PLAYLIST_SORT_TITLE); | |
1831 playlistwin_update_list(playlist); | |
1832 } | |
1833 | |
1834 void action_playlist_sort_by_artist(void) | |
1835 { | |
1836 Playlist *playlist = playlist_get_active(); | |
1837 | |
1838 playlist_sort(playlist, PLAYLIST_SORT_ARTIST); | |
1839 playlistwin_update_list(playlist); | |
1840 } | |
1841 | |
1842 void action_playlist_sort_by_full_path(void) | |
1843 { | |
1844 Playlist *playlist = playlist_get_active(); | |
1845 | |
1846 playlist_sort(playlist, PLAYLIST_SORT_PATH); | |
1847 playlistwin_update_list(playlist); | |
1848 } | |
1849 | |
1850 void action_playlist_sort_by_date(void) | |
1851 { | |
1852 Playlist *playlist = playlist_get_active(); | |
1853 | |
1854 playlist_sort(playlist, PLAYLIST_SORT_DATE); | |
1855 playlistwin_update_list(playlist); | |
1856 } | |
1857 | |
1858 void action_playlist_sort_by_filename(void) | |
1859 { | |
1860 Playlist *playlist = playlist_get_active(); | |
1861 | |
1862 playlist_sort(playlist, PLAYLIST_SORT_FILENAME); | |
1863 playlistwin_update_list(playlist); | |
1864 } | |
1865 | |
1866 void action_playlist_sort_selected_by_playlist_entry(void) | |
1867 { | |
1868 Playlist *playlist = playlist_get_active(); | |
1869 | |
1870 playlist_sort_selected(playlist, PLAYLIST_SORT_PLAYLIST); | |
1871 playlistwin_update_list(playlist); | |
1872 } | |
1873 | |
1874 void action_playlist_sort_selected_by_track_number(void) | |
1875 { | |
1876 Playlist *playlist = playlist_get_active(); | |
1877 | |
1878 playlist_sort_selected(playlist, PLAYLIST_SORT_TRACK); | |
1879 playlistwin_update_list(playlist); | |
1880 } | |
1881 | |
1882 void action_playlist_sort_selected_by_title(void) | |
1883 { | |
1884 Playlist *playlist = playlist_get_active(); | |
1885 | |
1886 playlist_sort_selected(playlist, PLAYLIST_SORT_TITLE); | |
1887 playlistwin_update_list(playlist); | |
1888 } | |
1889 | |
1890 void action_playlist_sort_selected_by_artist(void) | |
1891 { | |
1892 Playlist *playlist = playlist_get_active(); | |
1893 | |
1894 playlist_sort_selected(playlist, PLAYLIST_SORT_ARTIST); | |
1895 playlistwin_update_list(playlist); | |
1896 } | |
1897 | |
1898 void action_playlist_sort_selected_by_full_path(void) | |
1899 { | |
1900 Playlist *playlist = playlist_get_active(); | |
1901 | |
1902 playlist_sort_selected(playlist, PLAYLIST_SORT_PATH); | |
1903 playlistwin_update_list(playlist); | |
1904 } | |
1905 | |
1906 void action_playlist_sort_selected_by_date(void) | |
1907 { | |
1908 Playlist *playlist = playlist_get_active(); | |
1909 | |
1910 playlist_sort_selected(playlist, PLAYLIST_SORT_DATE); | |
1911 playlistwin_update_list(playlist); | |
1912 } | |
1913 | |
1914 void action_playlist_sort_selected_by_filename(void) | |
1915 { | |
1916 Playlist *playlist = playlist_get_active(); | |
1917 | |
1918 playlist_sort_selected(playlist, PLAYLIST_SORT_FILENAME); | |
1919 playlistwin_update_list(playlist); | |
1920 } | |
1921 | |
1922 void action_playlist_randomize_list(void) | |
1923 { | |
1924 Playlist *playlist = playlist_get_active(); | |
1925 | |
1926 playlist_random(playlist); | |
1927 playlistwin_update_list(playlist); | |
1928 } | |
1929 | |
1930 void action_playlist_reverse_list(void) | |
1931 { | |
1932 Playlist *playlist = playlist_get_active(); | |
1933 | |
1934 playlist_reverse(playlist); | |
1935 playlistwin_update_list(playlist); | |
1936 } | |
1937 | |
1938 void | |
1939 action_playlist_clear_queue(void) | |
1940 { | |
1941 playlist_clear_queue(playlist_get_active()); | |
1942 } | |
1943 | |
1944 void | |
1945 action_playlist_remove_unavailable(void) | |
1946 { | |
1947 playlist_remove_dead_files(playlist_get_active()); | |
1948 } | |
1949 | |
1950 void | |
1951 action_playlist_remove_dupes_by_title(void) | |
1952 { | |
1953 playlist_remove_duplicates(playlist_get_active(), PLAYLIST_DUPS_TITLE); | |
1954 } | |
1955 | |
1956 void | |
1957 action_playlist_remove_dupes_by_filename(void) | |
1958 { | |
1959 playlist_remove_duplicates(playlist_get_active(), PLAYLIST_DUPS_FILENAME); | |
1960 } | |
1961 | |
1962 void | |
1963 action_playlist_remove_dupes_by_full_path(void) | |
1964 { | |
1965 playlist_remove_duplicates(playlist_get_active(), PLAYLIST_DUPS_PATH); | |
1966 } | |
1967 | |
1968 void | |
1969 action_playlist_remove_all(void) | |
1970 { | |
1971 playlist_clear(playlist_get_active()); | |
1972 | |
1973 /* XXX -- should this really be coupled here? -nenolod */ | |
1974 mainwin_clear_song_info(); | |
1975 mainwin_set_info_text(); | |
1976 } | |
1977 | |
1978 void | |
1979 action_playlist_remove_selected(void) | |
1980 { | |
1981 playlist_delete(playlist_get_active(), FALSE); | |
1982 } | |
1983 | |
1984 void | |
1985 action_playlist_remove_unselected(void) | |
1986 { | |
1987 playlist_delete(playlist_get_active(), TRUE); | |
1988 } | |
1989 | |
1990 void | |
1991 action_playlist_add_files(void) | |
1992 { | |
2416
0fd7f4f969ad
[svn] integrated urldecode.* from libaudacious into audacious directory, made separate ui_fileopener.*
mf0102
parents:
2373
diff
changeset
|
1993 run_filebrowser(NO_PLAY_BUTTON); |
2313 | 1994 } |
1995 | |
1996 void add_medium(void); /* XXX */ | |
1997 | |
1998 void | |
1999 action_playlist_add_cd(void) | |
2000 { | |
2001 add_medium(); | |
2002 } | |
2003 | |
2004 void | |
2005 action_playlist_add_url(void) | |
2006 { | |
2007 mainwin_show_add_url_window(); | |
2008 } | |
2009 | |
2010 void | |
2011 action_playlist_new( void ) | |
2012 { | |
2013 Playlist *new_pl = playlist_new(); | |
2014 playlist_add_playlist(new_pl); | |
2015 playlist_select_playlist(new_pl); | |
2016 } | |
2017 | |
2018 void | |
2019 action_playlist_prev( void ) | |
2020 { | |
2021 playlist_select_prev(); | |
2022 } | |
2023 | |
2024 void | |
2025 action_playlist_next( void ) | |
2026 { | |
2027 playlist_select_next(); | |
2028 } | |
2029 | |
2030 void | |
2031 action_playlist_delete( void ) | |
2032 { | |
2033 playlist_remove_playlist( playlist_get_active() ); | |
2034 } | |
2035 | |
2036 void | |
2037 action_playlist_save_list(void) | |
2038 { | |
2039 Playlist *playlist = playlist_get_active(); | |
2040 | |
2041 playlistwin_select_playlist_to_save(playlist_get_current_name(playlist)); | |
2042 } | |
2043 | |
2044 void | |
2045 action_playlist_save_default_list(void) | |
2046 { | |
2047 Playlist *playlist = playlist_get_active(); | |
2048 | |
2049 playlist_save(playlist, bmp_paths[BMP_PATH_PLAYLIST_FILE]); | |
2050 } | |
2051 | |
2052 void | |
2053 action_playlist_load_list(void) | |
2054 { | |
2055 Playlist *playlist = playlist_get_active(); | |
2056 | |
2057 playlistwin_select_playlist_to_load(playlist_get_current_name(playlist)); | |
2058 } | |
2059 | |
2060 void | |
2061 action_playlist_refresh_list(void) | |
2062 { | |
2063 Playlist *playlist = playlist_get_active(); | |
2064 | |
2065 playlist_read_info_selection(playlist); | |
2066 playlistwin_update_list(playlist); | |
2067 } | |
2068 | |
2069 void | |
2070 action_open_list_manager(void) | |
2071 { | |
2072 playlist_manager_ui_show(); | |
2073 } | |
2074 | |
2075 void | |
2076 action_playlist_search_and_select(void) | |
2077 { | |
2078 playlistwin_select_search(); | |
2079 } | |
2080 | |
2081 void | |
2082 action_playlist_invert_selection(void) | |
2083 { | |
2084 playlistwin_inverse_selection(); | |
2085 } | |
2086 | |
2087 void | |
2088 action_playlist_select_none(void) | |
2089 { | |
2090 playlistwin_select_none(); | |
2091 } | |
2092 | |
2093 void | |
2094 action_playlist_select_all(void) | |
2095 { | |
2096 playlistwin_select_all(); | |
2097 } | |
2098 | |
2099 | |
2100 | |
2101 /* playlistwin_select_search callback functions | |
2102 placed here to avoid making the code messier :) */ | |
2103 void | |
2363 | 2104 playlistwin_select_search_cbt_cb(GtkWidget *called_cbt, gpointer other_cbt) |
2313 | 2105 { |
2363 | 2106 if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(called_cbt)) == TRUE) |
2107 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(other_cbt), FALSE); | |
2313 | 2108 return; |
2109 } | |
2110 | |
2111 static gboolean | |
2363 | 2112 playlistwin_select_search_kp_cb(GtkWidget *entry, GdkEventKey *event, |
2113 gpointer searchdlg_win) | |
2313 | 2114 { |
2115 switch (event->keyval) | |
2116 { | |
2117 case GDK_Return: | |
2363 | 2118 gtk_dialog_response(GTK_DIALOG(searchdlg_win), GTK_RESPONSE_ACCEPT); |
2313 | 2119 return TRUE; |
2120 default: | |
2121 return FALSE; | |
2122 } | |
2123 } | |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2124 |
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2125 |
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2126 /* fileinfopopup callback for playlistwin */ |
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2127 static gboolean |
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2128 playlistwin_fileinfopopup_probe(gpointer * filepopup_win) |
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2129 { |
2363 | 2130 gint x, y, pos; |
2131 TitleInput *tuple; | |
2132 static gint prev_x = 0, prev_y = 0, ctr = 0, prev_pos = -1; | |
2133 static gint shaded_pos = -1, shaded_prev_pos = -1; | |
2134 gboolean skip = FALSE; | |
2135 GdkWindow *win; | |
2136 Playlist *playlist = playlist_get_active(); | |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2137 |
2363 | 2138 win = gdk_window_at_pointer(NULL, NULL); |
2139 gdk_window_get_pointer(GDK_WINDOW(playlistwin->window), &x, &y, NULL); | |
2140 pos = playlist_list_get_playlist_position(playlistwin_list, x, y); | |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2141 |
2363 | 2142 if (win == NULL |
2143 || cfg.show_filepopup_for_tuple == FALSE | |
2144 || playlistwin_list->pl_tooltips == FALSE | |
2145 || pos != prev_pos | |
2146 || win != GDK_WINDOW(playlistwin->window)) | |
2147 { | |
2148 prev_pos = pos; | |
2149 ctr = 0; | |
2150 audacious_fileinfopopup_hide(GTK_WIDGET(filepopup_win), NULL); | |
2151 return TRUE; | |
2152 } | |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2153 |
2363 | 2154 if (prev_x == x && prev_y == y) |
2155 ctr++; | |
2156 else | |
2157 { | |
2158 ctr = 0; | |
2159 prev_x = x; | |
2160 prev_y = y; | |
2161 audacious_fileinfopopup_hide(GTK_WIDGET(filepopup_win), NULL); | |
2162 return TRUE; | |
2163 } | |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2164 |
2363 | 2165 if (playlistwin_is_shaded()) |
2166 { | |
2167 shaded_pos = playlist_get_position(playlist); | |
2168 if (shaded_prev_pos != shaded_pos) | |
2169 skip = TRUE; | |
2170 } | |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2171 |
2351
911743d27aba
[svn] - simplify and optimize the metadata tooltip trigger function
giacomo
parents:
2348
diff
changeset
|
2172 if (ctr >= cfg.filepopup_delay && (skip == TRUE || GTK_WIDGET_VISIBLE(GTK_WIDGET(filepopup_win)) != TRUE)) { |
2363 | 2173 if (pos == -1 && !playlistwin_is_shaded()) |
2174 { | |
2175 audacious_fileinfopopup_hide(GTK_WIDGET(filepopup_win), NULL); | |
2176 return TRUE; | |
2177 } | |
2178 /* shaded mode */ | |
2179 else | |
2180 { | |
2181 tuple = playlist_get_tuple(playlist, shaded_pos); | |
2182 audacious_fileinfopopup_hide(GTK_WIDGET(filepopup_win), NULL); | |
2183 audacious_fileinfopopup_show_from_tuple(GTK_WIDGET(filepopup_win), tuple); | |
2184 shaded_prev_pos = shaded_pos; | |
2185 } | |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2186 |
2363 | 2187 prev_pos = pos; |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2188 |
2363 | 2189 tuple = playlist_get_tuple(playlist, pos); |
2190 audacious_fileinfopopup_show_from_tuple(GTK_WIDGET(filepopup_win), tuple); | |
2191 } | |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2192 |
2363 | 2193 return TRUE; |
2348
564e8a1fe09a
[svn] - made a public API for fileinfopopup (popup that displays metadata, the same used in playlist); now plugins can include ui_fileinfopopup.h to display metadata popups (i.e. statusicon and libnotify plugins)
giacomo
parents:
2338
diff
changeset
|
2194 } |