Mercurial > audlegacy
diff src/audacious/ui_main.c @ 2313:3149d4b1a9a9 trunk
[svn] - objective-make autodepend fixes
- move all sourcecode into src/ and adjust Makefiles accordingly
author | nenolod |
---|---|
date | Fri, 12 Jan 2007 11:43:40 -0800 |
parents | |
children | d88558b0de0a |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/audacious/ui_main.c Fri Jan 12 11:43:40 2007 -0800 @@ -0,0 +1,3991 @@ +/* Audacious - Cross-platform multimedia player + * Copyright (C) 2005-2006 Audacious development team. + * + * BMP - Cross-platform multimedia player + * Copyright (C) 2003-2004 BMP development team. + * + * Based on XMMS: + * Copyright (C) 1998-2003 XMMS development team. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; under version 2 of the License. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + */ + +#ifdef HAVE_CONFIG_H +# include "config.h" +#endif + + +#include <glib.h> +#include <glib/gi18n.h> +#include <glib/gprintf.h> +#include <gtk/gtk.h> +#include <gtk/gtkmessagedialog.h> + +/* GDK including */ +#include "platform/smartinclude.h" + +#include <math.h> +#include <stdlib.h> +#include <string.h> + +#include <X11/Xlib.h> + +#include <sys/types.h> + +#if defined(USE_REGEX_ONIGURUMA) + #include <onigposix.h> +#elif defined(USE_REGEX_PCRE) + #include <pcreposix.h> +#else + #include <regex.h> +#endif + +#include "widgets/widgetcore.h" +#include "ui_main.h" +#include "icons-stock.h" + +#include "ui_manager.h" +#include "actions-mainwin.h" + +#include "main.h" + +#include "controlsocket.h" +#include "pluginenum.h" + +#include "ui_credits.h" +#include "dnd.h" +#include "dock.h" +#include "ui_equalizer.h" +#include "hints.h" +#include "input.h" +#include "ui_playlist.h" +#include "ui_preferences.h" +#include "ui_skinselector.h" +#include "genevent.h" +#include "playback.h" +#include "playlist.h" +#include "libaudacious/urldecode.h" +#include "util.h" +#include "visualization.h" +#include "libaudacious/configdb.h" + +static GTimeVal cb_time; /* click delay for tristate is defined by TRISTATE_THRESHOLD */ + +#define ITEM_SEPARATOR {"/-", NULL, NULL, 0, "<Separator>"} +#define TRISTATE_THRESHOLD 200 + +#define VOLSET_DISP_TIMES 5 + +enum { + MAINWIN_SEEK_REV = -1, + MAINWIN_SEEK_NIL, + MAINWIN_SEEK_FWD +}; + +enum { + MAINWIN_VIS_ACTIVE_MAINWIN, MAINWIN_VIS_ACTIVE_PLAYLISTWIN +}; + + +typedef struct _PlaybackInfo PlaybackInfo; + +struct _PlaybackInfo { + gchar *title; + gint bitrate; + gint frequency; + gint n_channels; +}; + + +GtkWidget *mainwin = NULL; +GtkWidget *err = NULL; /* an error dialog for miscellaneous error messages */ + +static GdkBitmap *nullmask; +static gint balance; + +GtkWidget *mainwin_jtf = NULL; +static GtkWidget *mainwin_jtt = NULL; + +gint seek_state = MAINWIN_SEEK_NIL; +gint seek_initial_pos = 0; + +GdkGC *mainwin_gc; +static GdkPixmap *mainwin_bg = NULL, *mainwin_bg_x2 = NULL; + +static PButton *mainwin_menubtn; +static PButton *mainwin_minimize, *mainwin_shade, *mainwin_close; + +static PButton *mainwin_rew, *mainwin_fwd; +static PButton *mainwin_eject; +static PButton *mainwin_play, *mainwin_pause, *mainwin_stop; + +TButton *mainwin_shuffle, *mainwin_repeat, *mainwin_eq, *mainwin_pl; +TextBox *mainwin_info; +TextBox *mainwin_stime_min, *mainwin_stime_sec; + +static TextBox *mainwin_rate_text, *mainwin_freq_text, + *mainwin_othertext; + +PlayStatus *mainwin_playstatus; + +Number *mainwin_minus_num, *mainwin_10min_num, *mainwin_min_num; +Number *mainwin_10sec_num, *mainwin_sec_num; + +static gboolean setting_volume = FALSE; + +Vis *active_vis; +Vis *mainwin_vis; +SVis *mainwin_svis; + +HSlider *mainwin_sposition = NULL; + +static MenuRow *mainwin_menurow; +static HSlider *mainwin_volume, *mainwin_balance, *mainwin_position; +static MonoStereo *mainwin_monostereo; +static SButton *mainwin_srew, *mainwin_splay, *mainwin_spause; +static SButton *mainwin_sstop, *mainwin_sfwd, *mainwin_seject, *mainwin_about; + +static GList *mainwin_wlist = NULL; + +static gint mainwin_timeout_id; + +G_LOCK_DEFINE_STATIC(mainwin_title); + +static gboolean mainwin_force_redraw = FALSE; +static gchar *mainwin_title_text = NULL; +static gboolean mainwin_info_text_locked = FALSE; + +static int ab_position_a = -1; +static int ab_position_b = -1; + +static PlaybackInfo playback_info = { NULL, 0, 0, 0 }; + + +static gint mainwin_idle_func(gpointer data); + +static void set_timer_mode_menu_cb(TimerMode mode); +static void set_timer_mode(TimerMode mode); + +static void mainwin_refresh_hints(void); + +void mainwin_position_motion_cb(gint pos); +void mainwin_position_release_cb(gint pos); + +void set_doublesize(gboolean doublesize); + + + +/* FIXME: placed here for now */ +void +playback_get_sample_params(gint * bitrate, + gint * frequency, + gint * n_channels) +{ + if (bitrate) + *bitrate = playback_info.bitrate; + + if (frequency) + *frequency = playback_info.frequency; + + if (n_channels) + *n_channels = playback_info.n_channels; +} + +static void +playback_set_sample_params(gint bitrate, + gint frequency, + gint n_channels) +{ + if (bitrate >= 0) + playback_info.bitrate = bitrate; + + if (frequency >= 0) + playback_info.frequency = frequency; + + if (n_channels >= 0) + playback_info.n_channels = n_channels; +} + +static void +mainwin_set_title_scroll(gboolean scroll) +{ + cfg.autoscroll = scroll; + textbox_set_scroll(mainwin_info, cfg.autoscroll); +} + + +void +mainwin_set_always_on_top(gboolean always) +{ + GtkAction *action = gtk_action_group_get_action( + toggleaction_group_others , "view always on top" ); + gtk_toggle_action_set_active( GTK_TOGGLE_ACTION(action) , always ); +} + +static void +mainwin_set_shape_mask(void) +{ + if (!cfg.player_visible) + return; + + if (cfg.doublesize == FALSE) + gtk_widget_shape_combine_mask(mainwin, + skin_get_mask(bmp_active_skin, + SKIN_MASK_MAIN), 0, 0); + else + gtk_widget_shape_combine_mask(mainwin, NULL, 0, 0); +} + +static void +mainwin_set_shade(gboolean shaded) +{ + GtkAction *action = gtk_action_group_get_action( + toggleaction_group_others , "roll up player" ); + gtk_toggle_action_set_active( GTK_TOGGLE_ACTION(action) , shaded ); +} + +static void +mainwin_set_shade_menu_cb(gboolean shaded) +{ + cfg.player_shaded = shaded; + + mainwin_set_shape_mask(); + + if (shaded) { + dock_shade(dock_window_list, GTK_WINDOW(mainwin), + MAINWIN_SHADED_HEIGHT * (cfg.doublesize + 1)); + + widget_show(WIDGET(mainwin_svis)); + vis_clear_data(mainwin_vis); + + widget_show(WIDGET(mainwin_srew)); + widget_show(WIDGET(mainwin_splay)); + widget_show(WIDGET(mainwin_spause)); + widget_show(WIDGET(mainwin_sstop)); + widget_show(WIDGET(mainwin_sfwd)); + widget_show(WIDGET(mainwin_seject)); + + textbox_set_scroll(mainwin_info, FALSE); + if (playback_get_playing()) + { + widget_show(WIDGET(mainwin_sposition)); + widget_show(WIDGET(mainwin_stime_min)); + widget_show(WIDGET(mainwin_stime_sec)); + } + else + { + widget_hide(WIDGET(mainwin_sposition)); + widget_hide(WIDGET(mainwin_stime_min)); + widget_hide(WIDGET(mainwin_stime_sec)); + } + + mainwin_shade->pb_ny = mainwin_shade->pb_py = 27; + } + else { + gint height = !bmp_active_skin->properties.mainwin_height ? MAINWIN_HEIGHT : + bmp_active_skin->properties.mainwin_height; + + dock_shade(dock_window_list, GTK_WINDOW(mainwin), height * (cfg.doublesize + 1)); + + widget_hide(WIDGET(mainwin_svis)); + svis_clear_data(mainwin_svis); + + widget_hide(WIDGET(mainwin_srew)); + widget_hide(WIDGET(mainwin_splay)); + widget_hide(WIDGET(mainwin_spause)); + widget_hide(WIDGET(mainwin_sstop)); + widget_hide(WIDGET(mainwin_sfwd)); + widget_hide(WIDGET(mainwin_seject)); + + widget_hide(WIDGET(mainwin_stime_min)); + widget_hide(WIDGET(mainwin_stime_sec)); + widget_hide(WIDGET(mainwin_sposition)); + + textbox_set_scroll(mainwin_info, cfg.autoscroll); + mainwin_shade->pb_ny = mainwin_shade->pb_py = 18; + } + + draw_main_window(TRUE); +} + +static void +mainwin_vis_set_active_vis(gint new_vis) +{ + active_vis = mainwin_vis; +} + +static void +mainwin_vis_set_refresh(RefreshRate rate) +{ + cfg.vis_refresh = rate; +} + +static void +mainwin_vis_set_afalloff(FalloffSpeed speed) +{ + cfg.analyzer_falloff = speed; +} + +static void +mainwin_vis_set_pfalloff(FalloffSpeed speed) +{ + cfg.peaks_falloff = speed; +} + +static void +mainwin_vis_set_analyzer_mode(AnalyzerMode mode) +{ + cfg.analyzer_mode = mode; +} + +static void +mainwin_vis_set_analyzer_type(AnalyzerType mode) +{ + cfg.analyzer_type = mode; +} + +void +mainwin_vis_set_type(VisType mode) +{ + GtkAction *action; + + switch ( mode ) + { + case VIS_ANALYZER: + action = gtk_action_group_get_action( + radioaction_group_vismode , "vismode analyzer" ); + break; + case VIS_SCOPE: + action = gtk_action_group_get_action( + radioaction_group_vismode , "vismode scope" ); + break; + case VIS_VOICEPRINT: + action = gtk_action_group_get_action( + radioaction_group_vismode , "vismode voiceprint" ); + break; + case VIS_OFF: + default: + action = gtk_action_group_get_action( + radioaction_group_vismode , "vismode off" ); + break; + } + + gtk_toggle_action_set_active( GTK_TOGGLE_ACTION(action) , TRUE ); +} + +static void +mainwin_vis_set_type_menu_cb(VisType mode) +{ + cfg.vis_type = mode; + + if (mode == VIS_OFF) { + if (cfg.player_shaded && cfg.player_visible) + svis_clear(mainwin_svis); + else + vis_clear(active_vis); + } + if (mode == VIS_ANALYZER || mode == VIS_SCOPE || mode == VIS_VOICEPRINT) { + vis_clear_data(active_vis); + svis_clear_data(mainwin_svis); + } +} + +static void +mainwin_menubtn_cb(void) +{ + gint x, y; + gtk_window_get_position(GTK_WINDOW(mainwin), &x, &y); + ui_manager_popup_menu_show(GTK_MENU(mainwin_general_menu), + x + 6 * (1 + cfg.doublesize), + y + MAINWIN_SHADED_HEIGHT * (1 + cfg.doublesize), + 1, GDK_CURRENT_TIME); +} + +void +mainwin_minimize_cb(void) +{ + if (!mainwin) + return; + + gtk_window_iconify(GTK_WINDOW(mainwin)); +} + +static void +mainwin_shade_toggle(void) +{ + mainwin_set_shade(!cfg.player_shaded); +} + +void +mainwin_quit_cb(void) +{ + GList *playlists = NULL, *playlists_top = NULL; + + gtk_widget_hide(equalizerwin); + gtk_widget_hide(playlistwin); + gtk_widget_hide(mainwin); + gdk_flush(); + + g_source_remove(mainwin_timeout_id); + + util_set_cursor(NULL); + + bmp_config_save(); + gtk_accel_map_save(bmp_paths[BMP_PATH_ACCEL_FILE]); + + ctrlsocket_cleanup(); + + plugin_system_cleanup(); + + playlist_stop_get_info_thread(); + + /* free and clear each playlist */ + playlists = playlist_get_playlists(); + playlists_top = playlists; + while ( playlists != NULL ) + { + playlist_clear((Playlist*)playlists->data); + playlist_free((Playlist*)playlists->data); + playlists = g_list_next(playlists); + } + g_list_free( playlists_top ); + + gtk_main_quit(); + + exit(EXIT_SUCCESS); +} + +static void +mainwin_destroy(GtkWidget * widget, gpointer data) +{ + mainwin_quit_cb(); +} + +static void +mainwin_draw_titlebar(gboolean focus) +{ + skin_draw_mainwin_titlebar(bmp_active_skin, mainwin_bg, mainwin_gc, + cfg.player_shaded, focus || !cfg.dim_titlebar); +} + +void +draw_main_window(gboolean force) +{ + GdkImage *img, *img2x; + GList *wl; + Widget *w; + gboolean redraw; + + if (!cfg.player_visible) + return; + + if (force) + mainwin_refresh_hints(); + + widget_list_lock(mainwin_wlist); + + if (force) { + if (!cfg.player_shaded) + skin_draw_pixmap(bmp_active_skin, mainwin_bg, mainwin_gc, + SKIN_MAIN, 0, 0, 0, 0, bmp_active_skin->properties.mainwin_width, + bmp_active_skin->properties.mainwin_height); + mainwin_draw_titlebar(gtk_window_has_toplevel_focus + (GTK_WINDOW(mainwin))); + } + + widget_list_draw(mainwin_wlist, &redraw, force); + + if (redraw || force) { + if (force) { + if (cfg.doublesize) { + img = gdk_drawable_get_image(mainwin_bg, 0, 0, bmp_active_skin->properties.mainwin_width, + cfg.player_shaded ? + MAINWIN_SHADED_HEIGHT : + bmp_active_skin->properties.mainwin_height); + img2x = create_dblsize_image(img); + gdk_draw_image(mainwin_bg_x2, mainwin_gc, img2x, 0, 0, + 0, 0, bmp_active_skin->properties.mainwin_width * 2, + cfg.player_shaded ? MAINWIN_SHADED_HEIGHT * + 2 : bmp_active_skin->properties.mainwin_height * 2); + g_object_unref(img2x); + g_object_unref(img); + } + + gdk_window_clear(mainwin->window); + + } + else { + for (wl = mainwin_wlist; wl; wl = g_list_next(wl)) { + w = WIDGET(wl->data); + + if (!w->redraw || !w->visible) + continue; + + if (w->x > bmp_active_skin->properties.mainwin_width || + w->y > bmp_active_skin->properties.mainwin_height) + continue; + + if (cfg.doublesize) { + gint width, height; + + width = w->x + w->width <= bmp_active_skin->properties.mainwin_width ? w->width : (w->width - ((w->x + w->width) - bmp_active_skin->properties.mainwin_width)); + height = w->y + w->height <= bmp_active_skin->properties.mainwin_width ? w->height : (w->height - ((w->y + w->height) - bmp_active_skin->properties.mainwin_height)); + + img = gdk_drawable_get_image(mainwin_bg, w->x, w->y, + width, height); + img2x = create_dblsize_image(img); + gdk_draw_image(mainwin_bg_x2, mainwin_gc, + img2x, 0, 0, w->x << 1, w->y << 1, + width << 1, height << 1); + g_object_unref(img2x); + g_object_unref(img); + gdk_window_clear_area(mainwin->window, w->x << 1, + w->y << 1, width << 1, + height << 1); + } + else + gdk_window_clear_area(mainwin->window, w->x, w->y, + w->width, w->height); + w->redraw = FALSE; + } + } + + gdk_flush(); + } + + widget_list_unlock(mainwin_wlist); +} + + +void +mainwin_set_info_text(void) +{ + gchar *text; + + if (mainwin_info_text_locked) + return; + + if ((text = input_get_info_text()) != NULL) { + textbox_set_text(mainwin_info, text); + g_free(text); + } + else if ((text = playlist_get_info_text(playlist_get_active())) != NULL) { + textbox_set_text(mainwin_info, text); + g_free(text); + } +} + +static gchar *mainwin_tb_old_text = NULL; + +void +mainwin_lock_info_text(const gchar * text) +{ + if (mainwin_info_text_locked != TRUE) + mainwin_tb_old_text = g_strdup(bmp_active_skin->properties.mainwin_othertext_is_status ? + mainwin_othertext->tb_text : mainwin_info->tb_text); + + mainwin_info_text_locked = TRUE; + textbox_set_text(bmp_active_skin->properties.mainwin_othertext_is_status ? + mainwin_othertext : mainwin_info, text); +} + +void +mainwin_release_info_text(void) +{ + mainwin_info_text_locked = FALSE; + + if (mainwin_tb_old_text != NULL) + { + textbox_set_text(bmp_active_skin->properties.mainwin_othertext_is_status ? + mainwin_othertext : mainwin_info, mainwin_tb_old_text); + g_free(mainwin_tb_old_text); + mainwin_tb_old_text = NULL; + } + else + mainwin_set_info_text(); /* XXX: best we can do */ +} + + +static gchar * +make_mainwin_title(const gchar * title) +{ + if (title) + return g_strdup_printf(_("%s - Audacious"), title); + else + return g_strdup(_("Audacious")); +} + +void +mainwin_set_song_title(const gchar * title) +{ + G_LOCK(mainwin_title); + g_free(mainwin_title_text); + mainwin_title_text = make_mainwin_title(title); + G_UNLOCK(mainwin_title); +} + +static void +mainwin_refresh_hints(void) +{ + if (bmp_active_skin && bmp_active_skin->properties.mainwin_othertext + == TRUE) + { + widget_hide(WIDGET(mainwin_rate_text)); + widget_hide(WIDGET(mainwin_freq_text)); + widget_hide(WIDGET(mainwin_monostereo)); + + if (bmp_active_skin->properties.mainwin_othertext_visible) + widget_show(WIDGET(mainwin_othertext)); + } + else + { + widget_show(WIDGET(mainwin_rate_text)); + widget_show(WIDGET(mainwin_freq_text)); + widget_show(WIDGET(mainwin_monostereo)); + widget_hide(WIDGET(mainwin_othertext)); + } + + /* positioning and size attributes */ + if (bmp_active_skin->properties.mainwin_vis_x && bmp_active_skin->properties.mainwin_vis_y) + widget_move(WIDGET(mainwin_vis), bmp_active_skin->properties.mainwin_vis_x, + bmp_active_skin->properties.mainwin_vis_y); + + if (bmp_active_skin->properties.mainwin_vis_width) + widget_resize(WIDGET(mainwin_vis), bmp_active_skin->properties.mainwin_vis_width, + mainwin_vis->vs_widget.height); + + if (bmp_active_skin->properties.mainwin_text_x && bmp_active_skin->properties.mainwin_text_y) + widget_move(WIDGET(mainwin_info), bmp_active_skin->properties.mainwin_text_x, + bmp_active_skin->properties.mainwin_text_y); + + if (bmp_active_skin->properties.mainwin_text_width) + widget_resize(WIDGET(mainwin_info), bmp_active_skin->properties.mainwin_text_width, + mainwin_info->tb_widget.height); + + if (bmp_active_skin->properties.mainwin_infobar_x && bmp_active_skin->properties.mainwin_infobar_y) + widget_move(WIDGET(mainwin_othertext), bmp_active_skin->properties.mainwin_infobar_x, + bmp_active_skin->properties.mainwin_infobar_y); + + if (bmp_active_skin->properties.mainwin_number_0_x && bmp_active_skin->properties.mainwin_number_0_y) + widget_move(WIDGET(mainwin_minus_num), bmp_active_skin->properties.mainwin_number_0_x, + bmp_active_skin->properties.mainwin_number_0_y); + + if (bmp_active_skin->properties.mainwin_number_1_x && bmp_active_skin->properties.mainwin_number_1_y) + widget_move(WIDGET(mainwin_10min_num), bmp_active_skin->properties.mainwin_number_1_x, + bmp_active_skin->properties.mainwin_number_1_y); + + if (bmp_active_skin->properties.mainwin_number_2_x && bmp_active_skin->properties.mainwin_number_2_y) + widget_move(WIDGET(mainwin_min_num), bmp_active_skin->properties.mainwin_number_2_x, + bmp_active_skin->properties.mainwin_number_2_y); + + if (bmp_active_skin->properties.mainwin_number_3_x && bmp_active_skin->properties.mainwin_number_3_y) + widget_move(WIDGET(mainwin_10sec_num), bmp_active_skin->properties.mainwin_number_3_x, + bmp_active_skin->properties.mainwin_number_3_y); + + if (bmp_active_skin->properties.mainwin_number_4_x && bmp_active_skin->properties.mainwin_number_4_y) + widget_move(WIDGET(mainwin_sec_num), bmp_active_skin->properties.mainwin_number_4_x, + bmp_active_skin->properties.mainwin_number_4_y); + + if (bmp_active_skin->properties.mainwin_playstatus_x && bmp_active_skin->properties.mainwin_playstatus_y) + widget_move(WIDGET(mainwin_playstatus), bmp_active_skin->properties.mainwin_playstatus_x, + bmp_active_skin->properties.mainwin_playstatus_y); + + if (bmp_active_skin->properties.mainwin_volume_x && bmp_active_skin->properties.mainwin_volume_y) + widget_move(WIDGET(mainwin_volume), bmp_active_skin->properties.mainwin_volume_x, + bmp_active_skin->properties.mainwin_volume_y); + + if (bmp_active_skin->properties.mainwin_balance_x && bmp_active_skin->properties.mainwin_balance_y) + widget_move(WIDGET(mainwin_balance), bmp_active_skin->properties.mainwin_balance_x, + bmp_active_skin->properties.mainwin_balance_y); + + if (bmp_active_skin->properties.mainwin_position_x && bmp_active_skin->properties.mainwin_position_y) + widget_move(WIDGET(mainwin_position), bmp_active_skin->properties.mainwin_position_x, + bmp_active_skin->properties.mainwin_position_y); + + if (bmp_active_skin->properties.mainwin_previous_x && bmp_active_skin->properties.mainwin_previous_y) + widget_move(WIDGET(mainwin_rew), bmp_active_skin->properties.mainwin_previous_x, + bmp_active_skin->properties.mainwin_previous_y); + + if (bmp_active_skin->properties.mainwin_play_x && bmp_active_skin->properties.mainwin_play_y) + widget_move(WIDGET(mainwin_play), bmp_active_skin->properties.mainwin_play_x, + bmp_active_skin->properties.mainwin_play_y); + + if (bmp_active_skin->properties.mainwin_pause_x && bmp_active_skin->properties.mainwin_pause_y) + widget_move(WIDGET(mainwin_pause), bmp_active_skin->properties.mainwin_pause_x, + bmp_active_skin->properties.mainwin_pause_y); + + if (bmp_active_skin->properties.mainwin_stop_x && bmp_active_skin->properties.mainwin_stop_y) + widget_move(WIDGET(mainwin_stop), bmp_active_skin->properties.mainwin_stop_x, + bmp_active_skin->properties.mainwin_stop_y); + + if (bmp_active_skin->properties.mainwin_next_x && bmp_active_skin->properties.mainwin_next_y) + widget_move(WIDGET(mainwin_fwd), bmp_active_skin->properties.mainwin_next_x, + bmp_active_skin->properties.mainwin_next_y); + + if (bmp_active_skin->properties.mainwin_eject_x && bmp_active_skin->properties.mainwin_eject_y) + widget_move(WIDGET(mainwin_eject), bmp_active_skin->properties.mainwin_eject_x, + bmp_active_skin->properties.mainwin_eject_y); + + if (bmp_active_skin->properties.mainwin_eqbutton_x && bmp_active_skin->properties.mainwin_eqbutton_y) + widget_move(WIDGET(mainwin_eq), bmp_active_skin->properties.mainwin_eqbutton_x, + bmp_active_skin->properties.mainwin_eqbutton_y); + + if (bmp_active_skin->properties.mainwin_plbutton_x && bmp_active_skin->properties.mainwin_plbutton_y) + widget_move(WIDGET(mainwin_pl), bmp_active_skin->properties.mainwin_plbutton_x, + bmp_active_skin->properties.mainwin_plbutton_y); + + if (bmp_active_skin->properties.mainwin_shuffle_x && bmp_active_skin->properties.mainwin_shuffle_y) + widget_move(WIDGET(mainwin_shuffle), bmp_active_skin->properties.mainwin_shuffle_x, + bmp_active_skin->properties.mainwin_shuffle_y); + + if (bmp_active_skin->properties.mainwin_repeat_x && bmp_active_skin->properties.mainwin_repeat_y) + widget_move(WIDGET(mainwin_repeat), bmp_active_skin->properties.mainwin_repeat_x, + bmp_active_skin->properties.mainwin_repeat_y); + + if (bmp_active_skin->properties.mainwin_about_x && bmp_active_skin->properties.mainwin_about_y) + widget_move(WIDGET(mainwin_about), bmp_active_skin->properties.mainwin_about_x, + bmp_active_skin->properties.mainwin_about_y); + + if (bmp_active_skin->properties.mainwin_minimize_x && bmp_active_skin->properties.mainwin_minimize_y) + widget_move(WIDGET(mainwin_minimize), cfg.player_shaded ? 244 : bmp_active_skin->properties.mainwin_minimize_x, + cfg.player_shaded ? 3 : bmp_active_skin->properties.mainwin_minimize_y); + + if (bmp_active_skin->properties.mainwin_shade_x && bmp_active_skin->properties.mainwin_shade_y) + widget_move(WIDGET(mainwin_shade), cfg.player_shaded ? 254 : bmp_active_skin->properties.mainwin_shade_x, + cfg.player_shaded ? 3 : bmp_active_skin->properties.mainwin_shade_y); + + if (bmp_active_skin->properties.mainwin_close_x && bmp_active_skin->properties.mainwin_close_y) + widget_move(WIDGET(mainwin_close), cfg.player_shaded ? 264 : bmp_active_skin->properties.mainwin_close_x, + cfg.player_shaded ? 3 : bmp_active_skin->properties.mainwin_close_y); + + /* visibility attributes */ + if (bmp_active_skin->properties.mainwin_menurow_visible) + widget_show(WIDGET(mainwin_menurow)); + else + widget_hide(WIDGET(mainwin_menurow)); + + if (bmp_active_skin->properties.mainwin_text_visible) + widget_show(WIDGET(mainwin_info)); + else + widget_hide(WIDGET(mainwin_info)); + + if (bmp_active_skin->properties.mainwin_othertext_visible) + widget_show(WIDGET(mainwin_othertext)); + else + widget_hide(WIDGET(mainwin_othertext)); + + if (bmp_active_skin->properties.mainwin_vis_visible) + widget_show(WIDGET(mainwin_vis)); + else + widget_hide(WIDGET(mainwin_vis)); + + /* window size, mainwinWidth && mainwinHeight properties */ + if (bmp_active_skin->properties.mainwin_height && bmp_active_skin->properties.mainwin_width) + { + gint width, height; + + gdk_window_get_size(mainwin->window, &width, &height); + + if (width == bmp_active_skin->properties.mainwin_width * (cfg.doublesize + 1) && + height == bmp_active_skin->properties.mainwin_height * (cfg.doublesize + 1)) + return; + + dock_window_resize(GTK_WINDOW(mainwin), cfg.player_shaded ? MAINWIN_SHADED_WIDTH * (cfg.doublesize + 1) : bmp_active_skin->properties.mainwin_width * (cfg.doublesize + 1), + cfg.player_shaded ? MAINWIN_SHADED_HEIGHT * (cfg.doublesize + 1) : bmp_active_skin->properties.mainwin_height * (cfg.doublesize + 1), + bmp_active_skin->properties.mainwin_width * (cfg.doublesize + 1), + bmp_active_skin->properties.mainwin_height * (cfg.doublesize + 1)); + + g_object_unref(mainwin_bg); + g_object_unref(mainwin_bg_x2); + mainwin_bg = gdk_pixmap_new(mainwin->window, + bmp_active_skin->properties.mainwin_width, + bmp_active_skin->properties.mainwin_height, -1); + mainwin_bg_x2 = gdk_pixmap_new(mainwin->window, + bmp_active_skin->properties.mainwin_width * 2, + bmp_active_skin->properties.mainwin_height * 2, -1); + mainwin_set_back_pixmap(); + widget_list_change_pixmap(mainwin_wlist, mainwin_bg); + gdk_flush(); + } +} + +void +mainwin_set_song_info(gint bitrate, + gint frequency, + gint n_channels) +{ + gchar text[512]; + gchar *title; + Playlist *playlist = playlist_get_active(); + + playback_set_sample_params(bitrate, frequency, n_channels); + + if (bitrate != -1) { + bitrate /= 1000; + + if (bitrate < 1000) { + /* Show bitrate in 1000s */ + g_snprintf(text, sizeof(text), "%3d", bitrate); + textbox_set_text(mainwin_rate_text, text); + } + else { + /* Show bitrate in 100,000s */ + g_snprintf(text, sizeof(text), "%2dH", bitrate / 100); + textbox_set_text(mainwin_rate_text, text); + } + } + else + textbox_set_text(mainwin_rate_text, _("VBR")); + + /* Show sampling frequency in kHz */ + g_snprintf(text, sizeof(text), "%2d", frequency / 1000); + textbox_set_text(mainwin_freq_text, text); + + monostereo_set_num_channels(mainwin_monostereo, n_channels); + + if (cfg.player_shaded) + { + widget_show(WIDGET(mainwin_stime_min)); + widget_show(WIDGET(mainwin_stime_sec)); + } + + widget_show(WIDGET(mainwin_minus_num)); + widget_show(WIDGET(mainwin_10min_num)); + widget_show(WIDGET(mainwin_min_num)); + widget_show(WIDGET(mainwin_10sec_num)); + widget_show(WIDGET(mainwin_sec_num)); + + if (!playback_get_paused() && mainwin_playstatus != NULL) + playstatus_set_status(mainwin_playstatus, STATUS_PLAY); + + if (playlist_get_current_length(playlist) != -1) { + if (cfg.player_shaded) + widget_show(WIDGET(mainwin_sposition)); + widget_show(WIDGET(mainwin_position)); + } + else { + widget_hide(WIDGET(mainwin_position)); + widget_hide(WIDGET(mainwin_sposition)); + mainwin_force_redraw = TRUE; + } + + if (bmp_active_skin && bmp_active_skin->properties.mainwin_othertext + == TRUE) + { + if (bitrate != -1) + g_snprintf(text, 512, "%d kbps, %0.1f kHz, %s", + bitrate, + (gfloat) frequency / 1000, + (n_channels > 1) ? _("stereo") : _("mono")); + else + g_snprintf(text, 512, "VBR, %0.1f kHz, %s", + (gfloat) frequency / 1000, + (n_channels > 1) ? _("stereo") : _("mono")); + + textbox_set_text(mainwin_othertext, text); + + widget_hide(WIDGET(mainwin_rate_text)); + widget_hide(WIDGET(mainwin_freq_text)); + widget_hide(WIDGET(mainwin_monostereo)); + + if (bmp_active_skin->properties.mainwin_othertext_visible) + widget_show(WIDGET(mainwin_othertext)); + } + else + { + widget_show(WIDGET(mainwin_rate_text)); + widget_show(WIDGET(mainwin_freq_text)); + widget_show(WIDGET(mainwin_monostereo)); + widget_hide(WIDGET(mainwin_othertext)); + } + + title = playlist_get_info_text(playlist); + mainwin_set_song_title(title); + g_free(title); +} + +void +mainwin_clear_song_info(void) +{ + if (!mainwin) + return; + + /* clear title */ + G_LOCK(mainwin_title); + g_free(mainwin_title_text); + mainwin_title_text = NULL; + G_UNLOCK(mainwin_title); + + /* clear sampling parameters */ + playback_set_sample_params(0, 0, 0); + + mainwin_position->hs_pressed = FALSE; + mainwin_sposition->hs_pressed = FALSE; + + /* clear sampling parameter displays */ + textbox_set_text(mainwin_rate_text, " "); + textbox_set_text(mainwin_freq_text, " "); + monostereo_set_num_channels(mainwin_monostereo, 0); + + if (mainwin_playstatus != NULL) + playstatus_set_status(mainwin_playstatus, STATUS_STOP); + + /* hide playback time */ + widget_hide(WIDGET(mainwin_minus_num)); + widget_hide(WIDGET(mainwin_10min_num)); + widget_hide(WIDGET(mainwin_min_num)); + widget_hide(WIDGET(mainwin_10sec_num)); + widget_hide(WIDGET(mainwin_sec_num)); + + widget_hide(WIDGET(mainwin_stime_min)); + widget_hide(WIDGET(mainwin_stime_sec)); + + widget_hide(WIDGET(mainwin_position)); + widget_hide(WIDGET(mainwin_sposition)); + + widget_hide(WIDGET(mainwin_othertext)); + + playlistwin_hide_timer(); + draw_main_window(TRUE); + + vis_clear(active_vis); +} + +void +mainwin_disable_seekbar(void) +{ + if (!mainwin) + return; + + /* + * We dont call draw_main_window() here so this will not + * remove them visually. It will only prevent us from sending + * any seek calls to the input plugin before the input plugin + * calls ->set_info(). + */ + widget_hide(WIDGET(mainwin_position)); + widget_hide(WIDGET(mainwin_sposition)); +} + +static gboolean +mainwin_mouse_button_release(GtkWidget * widget, + GdkEventButton * event, + gpointer callback_data) +{ + gdk_pointer_ungrab(GDK_CURRENT_TIME); + + /* + * The gdk_flush() is just for making sure that the pointer really + * gets ungrabbed before calling any button callbacks + * + */ + + gdk_flush(); + + if (dock_is_moving(GTK_WINDOW(mainwin))) { + dock_move_release(GTK_WINDOW(mainwin)); + draw_playlist_window(TRUE); + } + + if (mainwin_menurow->mr_doublesize_selected) { + event->x /= 2; + event->y /= 2; + } + + handle_release_cb(mainwin_wlist, widget, event); + + draw_main_window(FALSE); + + return FALSE; +} + +static gboolean +mainwin_motion(GtkWidget * widget, + GdkEventMotion * event, + gpointer callback_data) +{ + int x, y; + GdkModifierType state; + + if (event->is_hint != FALSE) + { + gdk_window_get_pointer(GDK_WINDOW(mainwin->window), + &x, &y, &state); + + /* If it's a hint, we had to query X, so override the + * information we we're given... it's probably useless... --nenolod + */ + event->x = x; + event->y = y; + event->state = state; + } + else + { + x = event->x; + y = event->y; + state = event->state; + } + if (cfg.doublesize) { + event->x /= 2; + event->y /= 2; + } + if (dock_is_moving(GTK_WINDOW(mainwin))) { + dock_move_motion(GTK_WINDOW(mainwin), event); + } + else { + handle_motion_cb(mainwin_wlist, widget, event); + draw_main_window(FALSE); + } + + gdk_flush(); + + return FALSE; +} + +static gboolean +inside_sensitive_widgets(gint x, gint y) +{ + return (widget_contains(WIDGET(mainwin_menubtn), x, y) + || widget_contains(WIDGET(mainwin_minimize), x, y) + || widget_contains(WIDGET(mainwin_shade), x, y) + || widget_contains(WIDGET(mainwin_close), x, y) + || widget_contains(WIDGET(mainwin_rew), x, y) + || widget_contains(WIDGET(mainwin_play), x, y) + || widget_contains(WIDGET(mainwin_pause), x, y) + || widget_contains(WIDGET(mainwin_stop), x, y) + || widget_contains(WIDGET(mainwin_fwd), x, y) + || widget_contains(WIDGET(mainwin_eject), x, y) + || widget_contains(WIDGET(mainwin_shuffle), x, y) + || widget_contains(WIDGET(mainwin_repeat), x, y) + || widget_contains(WIDGET(mainwin_pl), x, y) + || widget_contains(WIDGET(mainwin_eq), x, y) + || widget_contains(WIDGET(mainwin_info), x, y) + || widget_contains(WIDGET(mainwin_menurow), x, y) + || widget_contains(WIDGET(mainwin_volume), x, y) + || widget_contains(WIDGET(mainwin_balance), x, y) + || (widget_contains(WIDGET(mainwin_position), x, y) && + widget_is_visible(WIDGET(mainwin_position))) + || widget_contains(WIDGET(mainwin_minus_num), x, y) + || widget_contains(WIDGET(mainwin_10min_num), x, y) + || widget_contains(WIDGET(mainwin_min_num), x, y) + || widget_contains(WIDGET(mainwin_10sec_num), x, y) + || widget_contains(WIDGET(mainwin_sec_num), x, y) + || widget_contains(WIDGET(mainwin_vis), x, y) + || widget_contains(WIDGET(mainwin_minimize), x, y) + || widget_contains(WIDGET(mainwin_shade), x, y) + || widget_contains(WIDGET(mainwin_close), x, y) + || widget_contains(WIDGET(mainwin_menubtn), x, y) + || widget_contains(WIDGET(mainwin_sposition), x, y) + || widget_contains(WIDGET(mainwin_stime_min), x, y) + || widget_contains(WIDGET(mainwin_stime_sec), x, y) + || widget_contains(WIDGET(mainwin_srew), x, y) + || widget_contains(WIDGET(mainwin_splay), x, y) + || widget_contains(WIDGET(mainwin_spause), x, y) + || widget_contains(WIDGET(mainwin_sstop), x, y) + || widget_contains(WIDGET(mainwin_sfwd), x, y) + || widget_contains(WIDGET(mainwin_seject), x, y) + || widget_contains(WIDGET(mainwin_svis), x, y) + || widget_contains(WIDGET(mainwin_about), x, y)); +} + +void +mainwin_scrolled(GtkWidget *widget, GdkEventScroll *event, + gpointer callback_data) +{ + Playlist *playlist = playlist_get_active(); + + switch (event->direction) { + case GDK_SCROLL_UP: + mainwin_set_volume_diff(cfg.mouse_change); + break; + case GDK_SCROLL_DOWN: + mainwin_set_volume_diff(-cfg.mouse_change); + break; + case GDK_SCROLL_LEFT: + if (playlist_get_current_length(playlist) != -1) + playback_seek(CLAMP(playback_get_time() - 1000, + 0, playlist_get_current_length(playlist)) / 1000); + break; + case GDK_SCROLL_RIGHT: + if (playlist_get_current_length(playlist) != -1) + playback_seek(CLAMP(playback_get_time() + 1000, + 0, playlist_get_current_length(playlist)) / 1000); + break; + } +} + +static gboolean +mainwin_mouse_button_press(GtkWidget * widget, + GdkEventButton * event, + gpointer callback_data) +{ + + gboolean grab = TRUE; + + if (cfg.doublesize) { + /* + * A hack to make doublesize transparent to callbacks. + * We should make a copy of this data instead of + * tampering with the data we get from gtk+ + */ + event->x /= 2; + event->y /= 2; + } + + if (event->button == 1 && event->type == GDK_BUTTON_PRESS && + !inside_sensitive_widgets(event->x, event->y) && + (cfg.easy_move || event->y < 14)) { + if (0 && hint_move_resize_available()) { + hint_move_resize(mainwin, event->x_root, event->y_root, TRUE); + grab = FALSE; + } + else { + gtk_window_present(GTK_WINDOW(mainwin)); + dock_move_press(dock_window_list, GTK_WINDOW(mainwin), event, + TRUE); + } + } + else if (event->button == 1 && event->type == GDK_2BUTTON_PRESS && + event->y < 14 && !inside_sensitive_widgets(event->x, event->y)) { + mainwin_set_shade(!cfg.player_shaded); + if (dock_is_moving(GTK_WINDOW(mainwin))) + dock_move_release(GTK_WINDOW(mainwin)); + } + else if (event->button == 1 && event->type == GDK_2BUTTON_PRESS && + widget_contains(WIDGET(mainwin_info), event->x, event->y)) { + playlist_fileinfo_current(playlist_get_active()); + } + else { + handle_press_cb(mainwin_wlist, widget, event); + draw_main_window(FALSE); + } + + if ((event->button == 1) && event->type != GDK_2BUTTON_PRESS && + (widget_contains(WIDGET(mainwin_vis), event->x, event->y) || + widget_contains(WIDGET(mainwin_svis), event->x, event->y))) { + + cfg.vis_type++; + + if (cfg.vis_type > VIS_OFF) + cfg.vis_type = VIS_ANALYZER; + + mainwin_vis_set_type(cfg.vis_type); + } + + if (event->button == 3) { + if (widget_contains(WIDGET(mainwin_info), event->x, event->y)) { + ui_manager_popup_menu_show(GTK_MENU(mainwin_songname_menu), + event->x_root, event->y_root, + 3, event->time); + grab = FALSE; + } + else if (widget_contains(WIDGET(mainwin_vis), event->x, event->y) || + widget_contains(WIDGET(mainwin_svis), event->x, event->y)) { + ui_manager_popup_menu_show(GTK_MENU(mainwin_visualization_menu), event->x_root, + event->y_root, 3, event->time); + grab = FALSE; + } + else if ( (event->y > 70) && (event->x < 128) ) + { + + ui_manager_popup_menu_show(GTK_MENU(mainwin_playback_menu), + event->x_root, + event->y_root, 3, event->time); + grab = FALSE; + } else { + /* + * Pop up the main menu a few pixels down. + * This will avoid that anything is selected + * if one right-clicks to focus the window + * without raising it. + * + ***MD I think the above is stupid, people don't expect this + * + */ + ui_manager_popup_menu_show(GTK_MENU(mainwin_general_menu), + event->x_root, + event->y_root, 3, event->time); + grab = FALSE; + } + } + + if (event->button == 1) + { + if (widget_contains(WIDGET(mainwin_minus_num), event->x, event->y) || + widget_contains(WIDGET(mainwin_10min_num), event->x, event->y) || + widget_contains(WIDGET(mainwin_min_num), event->x, event->y) || + widget_contains(WIDGET(mainwin_10sec_num), event->x, event->y) || + widget_contains(WIDGET(mainwin_sec_num), event->x, event->y) || + widget_contains(WIDGET(mainwin_stime_min), event->x, event->y) || + widget_contains(WIDGET(mainwin_stime_sec), event->x, event->y)) + { + if (cfg.timer_mode == TIMER_ELAPSED) + set_timer_mode(TIMER_REMAINING); + else + set_timer_mode(TIMER_ELAPSED); + } + } + + if (grab) + gdk_pointer_grab(mainwin->window, FALSE, + GDK_BUTTON_MOTION_MASK | + GDK_BUTTON_RELEASE_MASK, + GDK_WINDOW(GDK_NONE), NULL, GDK_CURRENT_TIME); + + return FALSE; +} + +static gboolean +mainwin_focus_in(GtkWidget * window, + GdkEventFocus * event, + gpointer data) +{ + mainwin_menubtn->pb_allow_draw = TRUE; + mainwin_minimize->pb_allow_draw = TRUE; + mainwin_shade->pb_allow_draw = TRUE; + mainwin_close->pb_allow_draw = TRUE; + draw_main_window(TRUE); + + return TRUE; +} + + +static gboolean +mainwin_focus_out(GtkWidget * widget, + GdkEventFocus * event, + gpointer callback_data) +{ + mainwin_menubtn->pb_allow_draw = FALSE; + mainwin_minimize->pb_allow_draw = FALSE; + mainwin_shade->pb_allow_draw = FALSE; + mainwin_close->pb_allow_draw = FALSE; + draw_main_window(TRUE); + + return TRUE; +} + +static gboolean +mainwin_keypress(GtkWidget * grab_widget, + GdkEventKey * event, + gpointer data) +{ + Playlist *playlist = playlist_get_active(); + + switch (event->keyval) { + + case GDK_Up: + case GDK_KP_Up: + case GDK_KP_8: + mainwin_set_volume_diff(2); + break; + case GDK_Down: + case GDK_KP_Down: + case GDK_KP_2: + mainwin_set_volume_diff(-2); + break; + case GDK_Left: + case GDK_KP_Left: + case GDK_KP_7: + if (playlist_get_current_length(playlist) != -1) + playback_seek(CLAMP + (playback_get_time() - 5000, 0, + playlist_get_current_length(playlist)) / 1000); + break; + case GDK_Right: + case GDK_KP_Right: + case GDK_KP_9: + if (playlist_get_current_length(playlist) != -1) + playback_seek(CLAMP + (playback_get_time() + 5000, 0, + playlist_get_current_length(playlist)) / 1000); + break; + case GDK_KP_4: + playlist_prev(playlist); + break; + case GDK_KP_6: + playlist_next(playlist); + break; + case GDK_KP_Insert: + mainwin_jump_to_file(); + break; + case GDK_KP_5: + mainwin_play_pushed(); + break; + case GDK_Escape: + mainwin_minimize_cb(); + break; + default: + return FALSE; + } + + return TRUE; +} + +static void +mainwin_jump_to_time_cb(GtkWidget * widget, + GtkWidget * entry) +{ + guint min = 0, sec = 0, params; + gint time; + Playlist *playlist = playlist_get_active(); + + params = sscanf(gtk_entry_get_text(GTK_ENTRY(entry)), "%u:%u", + &min, &sec); + if (params == 2) + time = (min * 60) + sec; + else if (params == 1) + time = min; + else + return; + + if (playlist_get_current_length(playlist) > -1 && + time <= (playlist_get_current_length(playlist) / 1000)) + { + playback_seek(time); + gtk_widget_destroy(mainwin_jtt); + } +} + + +void +mainwin_jump_to_time(void) +{ + GtkWidget *vbox, *hbox_new, *hbox_total; + GtkWidget *time_entry, *label, *bbox, *jump, *cancel; + guint tindex; + gchar time_str[10]; + + if (!playback_get_playing()) { + report_error("JIT can't be launched when no track is being played.\n"); + return; + } + + if (mainwin_jtt) { + gtk_window_present(GTK_WINDOW(mainwin_jtt)); + return; + } + + mainwin_jtt = gtk_window_new(GTK_WINDOW_TOPLEVEL); + gtk_window_set_type_hint(GTK_WINDOW(mainwin_jtt), + GDK_WINDOW_TYPE_HINT_DIALOG); + + gtk_window_set_title(GTK_WINDOW(mainwin_jtt), _("Jump to Time")); + gtk_window_set_position(GTK_WINDOW(mainwin_jtt), GTK_WIN_POS_CENTER); + gtk_window_set_transient_for(GTK_WINDOW(mainwin_jtt), + GTK_WINDOW(mainwin)); + + g_signal_connect(mainwin_jtt, "destroy", + G_CALLBACK(gtk_widget_destroyed), &mainwin_jtt); + gtk_container_border_width(GTK_CONTAINER(mainwin_jtt), 10); + + vbox = gtk_vbox_new(FALSE, 5); + gtk_container_add(GTK_CONTAINER(mainwin_jtt), vbox); + + hbox_new = gtk_hbox_new(FALSE, 0); + gtk_box_pack_start(GTK_BOX(vbox), hbox_new, TRUE, TRUE, 5); + + time_entry = gtk_entry_new(); + gtk_box_pack_start(GTK_BOX(hbox_new), time_entry, FALSE, FALSE, 5); + g_signal_connect(time_entry, "activate", + G_CALLBACK(mainwin_jump_to_time_cb), time_entry); + + gtk_widget_set_size_request(time_entry, 70, -1); + label = gtk_label_new(_("minutes:seconds")); + gtk_box_pack_start(GTK_BOX(hbox_new), label, FALSE, FALSE, 5); + + hbox_total = gtk_hbox_new(FALSE, 0); + gtk_box_pack_start(GTK_BOX(vbox), hbox_total, TRUE, TRUE, 5); + gtk_widget_show(hbox_total); + + /* FIXME: Disable display of current track length. It's not + updated when track changes */ +#if 0 + label = gtk_label_new(_("Track length:")); + gtk_box_pack_start(GTK_BOX(hbox_total), label, FALSE, FALSE, 5); + + len = playlist_get_current_length() / 1000; + g_snprintf(time_str, sizeof(time_str), "%u:%2.2u", len / 60, len % 60); + label = gtk_label_new(time_str); + + gtk_box_pack_start(GTK_BOX(hbox_total), label, FALSE, FALSE, 10); +#endif + + bbox = gtk_hbutton_box_new(); + gtk_box_pack_start(GTK_BOX(vbox), bbox, TRUE, TRUE, 0); + gtk_button_box_set_layout(GTK_BUTTON_BOX(bbox), GTK_BUTTONBOX_END); + gtk_button_box_set_spacing(GTK_BUTTON_BOX(bbox), 5); + + cancel = gtk_button_new_from_stock(GTK_STOCK_CANCEL); + GTK_WIDGET_SET_FLAGS(cancel, GTK_CAN_DEFAULT); + gtk_container_add(GTK_CONTAINER(bbox), cancel); + g_signal_connect_swapped(cancel, "clicked", + G_CALLBACK(gtk_widget_destroy), mainwin_jtt); + + jump = gtk_button_new_from_stock(GTK_STOCK_JUMP_TO); + GTK_WIDGET_SET_FLAGS(jump, GTK_CAN_DEFAULT); + gtk_container_add(GTK_CONTAINER(bbox), jump); + g_signal_connect(jump, "clicked", + G_CALLBACK(mainwin_jump_to_time_cb), time_entry); + + tindex = playback_get_time() / 1000; + g_snprintf(time_str, sizeof(time_str), "%u:%2.2u", tindex / 60, + tindex % 60); + gtk_entry_set_text(GTK_ENTRY(time_entry), time_str); + + gtk_entry_select_region(GTK_ENTRY(time_entry), 0, strlen(time_str)); + + gtk_widget_show_all(mainwin_jtt); + + gtk_widget_grab_focus(time_entry); + gtk_widget_grab_default(jump); +} + +static void +change_song(guint pos) +{ + if (playback_get_playing()) + playback_stop(); + + playlist_set_position(playlist_get_active(), pos); + playback_initiate(); +} + +static void +mainwin_jump_to_file_jump(GtkTreeView * treeview) +{ + GtkTreeModel *model; + GtkTreeSelection *selection; + GtkTreeIter iter; + guint pos; + + model = gtk_tree_view_get_model(treeview); + selection = gtk_tree_view_get_selection(treeview); + + if (!gtk_tree_selection_get_selected(selection, NULL, &iter)) + return; + + gtk_tree_model_get(model, &iter, 0, &pos, -1); + + change_song(pos - 1); + + /* FIXME: should only hide window */ + gtk_widget_destroy(mainwin_jtf); + mainwin_jtf = NULL; +} + +static void +mainwin_jump_to_file_jump_cb(GtkTreeView * treeview, + gpointer data) +{ + mainwin_jump_to_file_jump(treeview); +} + +static void +mainwin_jump_to_file_set_queue_button_label(GtkButton * button, + guint pos) +{ + if (playlist_is_position_queued(playlist_get_active(), pos)) + gtk_button_set_label(button, _("Un_queue")); + else + gtk_button_set_label(button, _("_Queue")); +} + +static void +mainwin_jump_to_file_queue_cb(GtkButton * button, + gpointer data) +{ + GtkTreeView *treeview; + GtkTreeModel *model; + GtkTreeSelection *selection; + GtkTreeIter iter; + guint pos; + + treeview = GTK_TREE_VIEW(data); + model = gtk_tree_view_get_model(treeview); + selection = gtk_tree_view_get_selection(treeview); + + if (!gtk_tree_selection_get_selected(selection, NULL, &iter)) + return; + + gtk_tree_model_get(model, &iter, 0, &pos, -1); + + playlist_queue_position(playlist_get_active(), (pos - 1)); + + mainwin_jump_to_file_set_queue_button_label(button, (pos - 1)); +} + +static void +mainwin_jump_to_file_selection_changed_cb(GtkTreeSelection *treesel, + gpointer data) +{ + GtkTreeView *treeview; + GtkTreeModel *model; + GtkTreeSelection *selection; + GtkTreeIter iter; + guint pos; + + treeview = gtk_tree_selection_get_tree_view(treesel); + model = gtk_tree_view_get_model(treeview); + selection = gtk_tree_view_get_selection(treeview); + + if (!gtk_tree_selection_get_selected(selection, NULL, &iter)) + return; + + gtk_tree_model_get(model, &iter, 0, &pos, -1); + + mainwin_jump_to_file_set_queue_button_label(GTK_BUTTON(data), (pos - 1)); +} + +static gboolean +mainwin_jump_to_file_edit_keypress_cb(GtkWidget * object, + GdkEventKey * event, + gpointer data) +{ + switch (event->keyval) { + case GDK_Return: + if (gtk_im_context_filter_keypress (GTK_ENTRY (object)->im_context, event)) { + GTK_ENTRY (object)->need_im_reset = TRUE; + return TRUE; + } else { + mainwin_jump_to_file_jump(GTK_TREE_VIEW(data)); + return TRUE; + } + default: + return FALSE; + } +} + +static gboolean +mainwin_jump_to_file_keypress_cb(GtkWidget * object, + GdkEventKey * event, + gpointer data) +{ + switch (event->keyval) { + case GDK_Escape: + /* FIXME: show only hide window */ + gtk_widget_destroy(mainwin_jtf); + mainwin_jtf = NULL; + return TRUE; + case GDK_KP_Enter: + mainwin_jump_to_file_queue_cb(NULL, data); + return TRUE; + default: + return FALSE; + }; + + return FALSE; +} + +static gboolean +mainwin_jump_to_file_match(const gchar * song, GSList *regex_list) +{ + gint i = 0; + gboolean rv = TRUE; + + if ( song == NULL ) + return FALSE; + + for ( ; regex_list ; regex_list = g_slist_next(regex_list) ) + { + regex_t *regex = regex_list->data; + if ( regexec( regex , song , 0 , NULL , 0 ) != 0 ) + { + rv = FALSE; + break; + } + } + + return rv; +} + +/* FIXME: Clear the entry when the list gets updated */ +static void +mainwin_update_jtf(GtkWidget * widget, gpointer user_data) +{ + /* FIXME: Is not in sync with playlist due to delayed extinfo + * reading */ + guint row; + GList *playlist_glist; + gchar *desc_buf = NULL; + gchar *row_str; + GtkTreeIter iter; + GtkTreeSelection *selection; + Playlist *playlist; + + GtkTreeModel *store; + + if (!mainwin_jtf) + return; + + store = gtk_tree_view_get_model(GTK_TREE_VIEW(user_data)); + gtk_list_store_clear(GTK_LIST_STORE(store)); + + row = 1; + playlist = playlist_get_active(); + for (playlist_glist = playlist->entries; playlist_glist; + playlist_glist = g_list_next(playlist_glist)) { + PlaylistEntry *entry = PLAYLIST_ENTRY(playlist_glist->data); + + if (entry->title) + desc_buf = g_strdup(entry->title); + else if (strchr(entry->filename, '/')) + desc_buf = str_to_utf8(strrchr(entry->filename, '/') + 1); + else + desc_buf = str_to_utf8(entry->filename); + + gtk_list_store_append(GTK_LIST_STORE(store), &iter); + gtk_list_store_set(GTK_LIST_STORE(store), &iter, + 0, row, 1, desc_buf, -1); + row++; + + if(desc_buf) { + g_free(desc_buf); + desc_buf = NULL; + } + } + + gtk_tree_model_get_iter_first(GTK_TREE_MODEL(store), &iter); + selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(user_data)); + gtk_tree_selection_select_iter(selection, &iter); +} + +static void +mainwin_jump_to_file_edit_cb(GtkEntry * entry, gpointer user_data) +{ + GtkTreeView *treeview = GTK_TREE_VIEW(user_data); + GtkTreeSelection *selection; + GtkTreeIter iter; + + GtkListStore *store; + + guint song_index = 0; + gchar **words; + GList *playlist_glist; + Playlist *playlist; + + gboolean match = FALSE; + + GSList *regex_list = NULL, *regex_list_tmp = NULL; + gint i = -1; + + /* Chop the key string into ' '-separated key regex-pattern strings */ + words = g_strsplit(gtk_entry_get_text(entry), " ", 0); + + /* create a list of regex using the regex-pattern strings */ + while ( words[++i] != NULL ) + { + regex_t *regex = g_malloc(sizeof(regex_t)); + #if defined(USE_REGEX_PCRE) + if ( regcomp( regex , words[i] , REG_NOSUB | REG_ICASE | REG_UTF8 ) == 0 ) + #else + if ( regcomp( regex , words[i] , REG_NOSUB | REG_ICASE ) == 0 ) + #endif + regex_list = g_slist_append( regex_list , regex ); + } + + /* FIXME: Remove the connected signals before clearing + * (row-selected will still eventually arrive once) */ + store = GTK_LIST_STORE(gtk_tree_view_get_model(treeview)); + /* detach model from treeview */ + g_object_ref( store ); + gtk_tree_view_set_model( GTK_TREE_VIEW(treeview) , NULL ); + + gtk_list_store_clear(store); + + playlist = playlist_get_active(); + + PLAYLIST_LOCK(playlist->mutex); + + for (playlist_glist = playlist->entries; playlist_glist; + playlist_glist = g_list_next(playlist_glist)) { + + PlaylistEntry *entry = PLAYLIST_ENTRY(playlist_glist->data); + const gchar *title; + gchar *filename = NULL; + + title = entry->title; + if (!title) { + filename = str_to_utf8(entry->filename); + + if (strchr(filename, '/')) + title = strrchr(filename, '/') + 1; + else + title = filename; + } + + /* Compare the reg.expressions to the string - if all the + regexp in regex_list match, add to the ListStore */ + + /* + * FIXME: The search string should be adapted to the + * current display setting, e.g. if the user has set it to + * "%p - %t" then build the match string like that too, or + * even better, search for each of the tags seperatly. + * + * In any case the string to match should _never_ contain + * something the user can't actually see in the playlist. + */ + if (regex_list != NULL) + match = mainwin_jump_to_file_match(title, regex_list); + else + match = TRUE; + + if (match) { + gtk_list_store_append(store, &iter); + gtk_list_store_set(store, &iter, 0, song_index + 1 , 1, title, -1); + } + + song_index++; + if (filename) { + g_free(filename); + filename = NULL; + } + } + + PLAYLIST_UNLOCK(playlist->mutex); + + /* attach the model again to the treeview */ + gtk_tree_view_set_model( GTK_TREE_VIEW(treeview) , GTK_TREE_MODEL(store) ); + g_object_unref( store ); + + if ( regex_list != NULL ) + { + regex_list_tmp = regex_list; + while ( regex_list != NULL ) + { + regex_t *regex = regex_list->data; + regfree( regex ); + regex_list = g_slist_next(regex_list); + } + g_slist_free( regex_list_tmp ); + } + g_strfreev(words); + + if (gtk_tree_model_get_iter_first(GTK_TREE_MODEL(store), &iter)) { + selection = gtk_tree_view_get_selection(treeview); + gtk_tree_selection_select_iter(selection, &iter); + } +} + +void +mainwin_jump_to_file(void) +{ + GtkWidget *scrollwin; + GtkWidget *vbox, *bbox, *sep; + GtkWidget *jump, *queue, *cancel; + GtkWidget *rescan, *edit; + GtkWidget *search_label, *hbox; + GList *playlist_glist; + Playlist *playlist; + gchar *desc_buf = NULL; + guint row; + + GtkWidget *treeview; + GtkListStore *jtf_store; + + GtkTreeIter iter; + GtkCellRenderer *renderer; + GtkTreeViewColumn *column; + + if (mainwin_jtf) { + gtk_window_present(GTK_WINDOW(mainwin_jtf)); + return; + } + + #if defined(USE_REGEX_ONIGURUMA) + /* set encoding for Oniguruma regex to UTF-8 */ + reg_set_encoding( REG_POSIX_ENCODING_UTF8 ); + onig_set_default_syntax( ONIG_SYNTAX_POSIX_BASIC ); + #endif + + mainwin_jtf = gtk_window_new(GTK_WINDOW_TOPLEVEL); + gtk_window_set_type_hint(GTK_WINDOW(mainwin_jtf), + GDK_WINDOW_TYPE_HINT_DIALOG); + + gtk_window_set_title(GTK_WINDOW(mainwin_jtf), _("Jump to Track")); + + gtk_window_set_position(GTK_WINDOW(mainwin_jtf), GTK_WIN_POS_CENTER); + g_signal_connect(mainwin_jtf, "destroy", + G_CALLBACK(gtk_widget_destroyed), &mainwin_jtf); + + gtk_container_border_width(GTK_CONTAINER(mainwin_jtf), 10); + gtk_window_set_default_size(GTK_WINDOW(mainwin_jtf), 550, 350); + + vbox = gtk_vbox_new(FALSE, 5); + gtk_container_add(GTK_CONTAINER(mainwin_jtf), vbox); + + jtf_store = gtk_list_store_new(2, G_TYPE_UINT, G_TYPE_STRING); + treeview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(jtf_store)); + g_object_unref(jtf_store); + + gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(treeview), TRUE); + + column = gtk_tree_view_column_new(); + gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(treeview), FALSE); + gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE); + + renderer = gtk_cell_renderer_text_new(); + gtk_tree_view_column_pack_start(column, renderer, FALSE); + gtk_tree_view_column_set_attributes(column, renderer, "text", 0, NULL); + gtk_tree_view_column_set_spacing(column, 4); + + renderer = gtk_cell_renderer_text_new(); + gtk_tree_view_column_pack_start(column, renderer, FALSE); + gtk_tree_view_column_set_attributes(column, renderer, "text", 1, NULL); + gtk_tree_view_column_set_spacing(column, 4); + gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column); + + gtk_tree_view_set_search_column(GTK_TREE_VIEW(treeview), 1); + + g_signal_connect(treeview, "row-activated", + G_CALLBACK(mainwin_jump_to_file_jump), NULL); + + hbox = gtk_hbox_new(FALSE, 3); + gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 3); + + search_label = gtk_label_new(_("Filter: ")); + gtk_label_set_markup_with_mnemonic(GTK_LABEL(search_label), _("_Filter:")); + gtk_box_pack_start(GTK_BOX(hbox), search_label, FALSE, FALSE, 0); + + edit = gtk_entry_new(); + gtk_entry_set_editable(GTK_ENTRY(edit), TRUE); + gtk_label_set_mnemonic_widget(GTK_LABEL(search_label), edit); + g_signal_connect(edit, "changed", + G_CALLBACK(mainwin_jump_to_file_edit_cb), treeview); + + g_signal_connect(edit, "key_press_event", + G_CALLBACK(mainwin_jump_to_file_edit_keypress_cb), treeview); + + g_signal_connect(mainwin_jtf, "key_press_event", + G_CALLBACK(mainwin_jump_to_file_keypress_cb), treeview); + + gtk_box_pack_start(GTK_BOX(hbox), edit, TRUE, TRUE, 3); + + scrollwin = gtk_scrolled_window_new(NULL, NULL); + gtk_container_add(GTK_CONTAINER(scrollwin), treeview); + gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrollwin), + GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS); + gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrollwin), + GTK_SHADOW_IN); + gtk_box_pack_start(GTK_BOX(vbox), scrollwin, TRUE, TRUE, 0); + + sep = gtk_hseparator_new(); + gtk_box_pack_start(GTK_BOX(vbox), sep, FALSE, FALSE, 0); + + bbox = gtk_hbutton_box_new(); + gtk_button_box_set_layout(GTK_BUTTON_BOX(bbox), GTK_BUTTONBOX_END); + gtk_button_box_set_spacing(GTK_BUTTON_BOX(bbox), 5); + gtk_box_pack_start(GTK_BOX(vbox), bbox, FALSE, FALSE, 0); + + queue = gtk_button_new_with_mnemonic(_("_Queue")); + gtk_box_pack_start(GTK_BOX(bbox), queue, FALSE, FALSE, 0); + GTK_WIDGET_SET_FLAGS(queue, GTK_CAN_DEFAULT); + g_signal_connect(queue, "clicked", + G_CALLBACK(mainwin_jump_to_file_queue_cb), + treeview); + g_signal_connect(gtk_tree_view_get_selection(GTK_TREE_VIEW(treeview)), "changed", + G_CALLBACK(mainwin_jump_to_file_selection_changed_cb), + queue); + + rescan = gtk_button_new_from_stock(GTK_STOCK_REFRESH); + gtk_box_pack_start(GTK_BOX(bbox), rescan, FALSE, FALSE, 0); + g_signal_connect(rescan, "clicked", + G_CALLBACK(mainwin_update_jtf), treeview); + GTK_WIDGET_SET_FLAGS(rescan, GTK_CAN_DEFAULT); + gtk_widget_grab_default(rescan); + + jump = gtk_button_new_from_stock(GTK_STOCK_JUMP_TO); + gtk_box_pack_start(GTK_BOX(bbox), jump, FALSE, FALSE, 0); + + g_signal_connect_swapped(jump, "clicked", + G_CALLBACK(mainwin_jump_to_file_jump_cb), + treeview); + + GTK_WIDGET_SET_FLAGS(jump, GTK_CAN_DEFAULT); + gtk_widget_grab_default(jump); + + cancel = gtk_button_new_from_stock(GTK_STOCK_CLOSE); + gtk_box_pack_start(GTK_BOX(bbox), cancel, FALSE, FALSE, 0); + g_signal_connect_swapped(cancel, "clicked", + G_CALLBACK(gtk_widget_destroy), + mainwin_jtf); + GTK_WIDGET_SET_FLAGS(cancel, GTK_CAN_DEFAULT); + + gtk_list_store_clear(jtf_store); + + row = 1; + + playlist = playlist_get_active(); + + PLAYLIST_LOCK(playlist->mutex); + + for (playlist_glist = playlist->entries; playlist_glist; + playlist_glist = g_list_next(playlist_glist)) { + + PlaylistEntry *entry = PLAYLIST_ENTRY(playlist_glist->data); + + if (entry->title) + desc_buf = g_strdup(entry->title); + else if (strchr(entry->filename, '/')) + desc_buf = str_to_utf8(strrchr(entry->filename, '/') + 1); + else + desc_buf = str_to_utf8(entry->filename); + + gtk_list_store_append(GTK_LIST_STORE(jtf_store), &iter); + gtk_list_store_set(GTK_LIST_STORE(jtf_store), &iter, + 0, row, 1, desc_buf, -1); + row++; + + if (desc_buf) { + g_free(desc_buf); + desc_buf = NULL; + } + } + + PLAYLIST_UNLOCK(playlist->mutex); + + gtk_widget_show_all(mainwin_jtf); +} + +static gboolean +mainwin_configure(GtkWidget * window, + GdkEventConfigure * event, + gpointer data) +{ + if (!GTK_WIDGET_VISIBLE(window)) + return FALSE; + + if (cfg.show_wm_decorations) + gdk_window_get_root_origin(window->window, + &cfg.player_x, &cfg.player_y); + else + gdk_window_get_deskrelative_origin(window->window, + &cfg.player_x, &cfg.player_y); + return FALSE; +} + +void +mainwin_set_back_pixmap(void) +{ + if (cfg.doublesize) + gdk_window_set_back_pixmap(mainwin->window, mainwin_bg_x2, 0); + else + gdk_window_set_back_pixmap(mainwin->window, mainwin_bg, 0); + gdk_window_clear(mainwin->window); +} + +/* + * Rewritten 09/13/06: + * + * Remove all of this flaky iter/sourcelist/strsplit stuff. + * All we care about is the filepath. + * + * We can figure this out and easily pass it to xmms_urldecode_plain(). + * - nenolod + */ +void +mainwin_drag_data_received(GtkWidget * widget, + GdkDragContext * context, + gint x, + gint y, + GtkSelectionData * selection_data, + guint info, + guint time, + gpointer user_data) +{ + Playlist *playlist = playlist_get_active(); + + g_return_if_fail(selection_data != NULL); + g_return_if_fail(selection_data->data != NULL); + + if (str_has_prefix_nocase((gchar *) selection_data->data, "fonts:///")) + { + gchar *path = (gchar *) selection_data->data + 9; /* skip fonts:/// */ + gchar *decoded = xmms_urldecode_plain(path); + + cfg.playlist_font = g_strconcat(decoded, strrchr(cfg.playlist_font, ' '), NULL); + playlist_list_set_font(cfg.playlist_font); + playlistwin_update_list(playlist); + + g_free(decoded); + + return; + } + + playlist_clear(playlist); + playlist_add_url(playlist, (gchar *) selection_data->data); + playback_initiate(); +} + +static void +on_add_url_add_clicked(GtkWidget * widget, + GtkWidget * entry) +{ + const gchar *text = gtk_entry_get_text(GTK_ENTRY(entry)); + if (text && *text) + playlist_add_url(playlist_get_active(), text); +} + +static void +on_add_url_ok_clicked(GtkWidget * widget, + GtkWidget * entry) +{ + Playlist *playlist = playlist_get_active(); + + const gchar *text = gtk_entry_get_text(GTK_ENTRY(entry)); + if (text && *text) + { + playlist_clear(playlist); + playlist_add_url(playlist, text); + playback_initiate(); + } +} + +void +mainwin_show_add_url_window(void) +{ + static GtkWidget *url_window = NULL; + + if (!url_window) { + url_window = + util_add_url_dialog_new(_("Enter location to play:"), + G_CALLBACK(on_add_url_ok_clicked), + G_CALLBACK(on_add_url_add_clicked)); + gtk_window_set_transient_for(GTK_WINDOW(url_window), + GTK_WINDOW(mainwin)); + g_signal_connect(url_window, "destroy", + G_CALLBACK(gtk_widget_destroyed), + &url_window); + } + + gtk_window_present(GTK_WINDOW(url_window)); +} + +static void +check_set( GtkActionGroup * action_group , + const gchar * action_name , + gboolean is_on ) +{ + /* check_set noew uses gtkaction */ + GtkAction *action = gtk_action_group_get_action( action_group , action_name ); + if ( action != NULL ) + gtk_toggle_action_set_active( GTK_TOGGLE_ACTION(action) , is_on ); + return; +} + +void +mainwin_eject_pushed(void) +{ + util_run_filebrowser(PLAY_BUTTON); +} + +void +mainwin_rev_pushed(void) +{ + g_get_current_time(&cb_time); + + seek_initial_pos = hslider_get_position(mainwin_position); + seek_state = MAINWIN_SEEK_REV; +} + +void +mainwin_rev_release(void) +{ + GTimeVal now_time; + GTimeVal delta_time; + gulong now_dur; + + g_get_current_time(&now_time); + + delta_time.tv_usec = now_time.tv_usec - cb_time.tv_usec; + delta_time.tv_sec = now_time.tv_sec - cb_time.tv_sec; + + now_dur = labs((delta_time.tv_sec * 1000) + (glong) (delta_time.tv_usec / 1000)); + + if ( now_dur <= TRISTATE_THRESHOLD ) + { + /* interpret as 'skip to previous song' */ + playlist_prev(playlist_get_active()); + } + else + { + /* interpret as 'seek' */ + mainwin_position_release_cb( hslider_get_position(mainwin_position) ); + } + + seek_state = MAINWIN_SEEK_NIL; +} + +void +mainwin_fwd_pushed(void) +{ + g_get_current_time(&cb_time); + + seek_initial_pos = hslider_get_position(mainwin_position); + seek_state = MAINWIN_SEEK_FWD; +} + +void +mainwin_fwd_release(void) +{ + GTimeVal now_time; + GTimeVal delta_time; + gulong now_dur; + + g_get_current_time(&now_time); + + delta_time.tv_usec = now_time.tv_usec - cb_time.tv_usec; + delta_time.tv_sec = now_time.tv_sec - cb_time.tv_sec; + + now_dur = labs((delta_time.tv_sec * 1000) + (glong) (delta_time.tv_usec / 1000)); + + if ( now_dur <= TRISTATE_THRESHOLD ) + { + /* interpret as 'skip to previous song' */ + playlist_next(playlist_get_active()); + } + else + { + /* interpret as 'seek' */ + mainwin_position_release_cb( hslider_get_position(mainwin_position) ); + } + + seek_state = MAINWIN_SEEK_NIL; +} + +void +mainwin_play_pushed(void) +{ + if (ab_position_a != -1) + playback_seek(ab_position_a / 1000); + if (playback_get_paused()) { + playback_pause(); + return; + } + + if (playlist_get_length(playlist_get_active())) + playback_initiate(); + else + mainwin_eject_pushed(); +} + +void +mainwin_stop_pushed(void) +{ + ip_data.stop = TRUE; + mainwin_clear_song_info(); + playback_stop(); + ip_data.stop = FALSE; +} + +void +mainwin_shuffle_pushed(gboolean toggled) +{ + check_set( toggleaction_group_others , "playback shuffle" , toggled ); +} + +void +mainwin_repeat_pushed(gboolean toggled) +{ + check_set( toggleaction_group_others , "playback repeat" , toggled ); +} + +void +mainwin_pl_pushed(gboolean toggled) +{ + if (toggled) + playlistwin_show(); + else + playlistwin_hide(); +} + +gint +mainwin_spos_frame_cb(gint pos) +{ + if (mainwin_sposition) { + if (pos < 6) + mainwin_sposition->hs_knob_nx = mainwin_sposition->hs_knob_px = + 17; + else if (pos < 9) + mainwin_sposition->hs_knob_nx = mainwin_sposition->hs_knob_px = + 20; + else + mainwin_sposition->hs_knob_nx = mainwin_sposition->hs_knob_px = + 23; + } + return 1; +} + +void +mainwin_spos_motion_cb(gint pos) +{ + gint time; + gchar *time_msg; + Playlist *playlist = playlist_get_active(); + + pos--; + + time = ((playlist_get_current_length(playlist) / 1000) * pos) / 12; + + if (cfg.timer_mode == TIMER_REMAINING) { + time = (playlist_get_current_length(playlist) / 1000) - time; + time_msg = g_strdup_printf("-%2.2d", time / 60); + textbox_set_text(mainwin_stime_min, time_msg); + g_free(time_msg); + } + else { + time_msg = g_strdup_printf(" %2.2d", time / 60); + textbox_set_text(mainwin_stime_min, time_msg); + g_free(time_msg); + } + + time_msg = g_strdup_printf("%2.2d", time % 60); + textbox_set_text(mainwin_stime_sec, time_msg); + g_free(time_msg); +} + +void +mainwin_spos_release_cb(gint pos) +{ + playback_seek(((playlist_get_current_length(playlist_get_active()) / 1000) * + (pos - 1)) / 12); +} + +void +mainwin_position_motion_cb(gint pos) +{ + gint length, time; + gchar *seek_msg; + + length = playlist_get_current_length(playlist_get_active()) / 1000; + time = (length * pos) / 219; + seek_msg = g_strdup_printf(_("SEEK TO: %d:%-2.2d/%d:%-2.2d (%d%%)"), + time / 60, time % 60, + length / 60, length % 60, + (length != 0) ? (time * 100) / length : 0); + mainwin_lock_info_text(seek_msg); + g_free(seek_msg); +} + +void +mainwin_position_release_cb(gint pos) +{ + gint length, time; + + length = playlist_get_current_length(playlist_get_active()) / 1000; + time = (length * pos) / 219; + playback_seek(time); + mainwin_release_info_text(); +} + +gint +mainwin_volume_frame_cb(gint pos) +{ + return (gint) rint((pos / 52.0) * 28); +} + +void +mainwin_adjust_volume_motion(gint v) +{ + gchar *volume_msg; + + setting_volume = TRUE; + + volume_msg = g_strdup_printf(_("VOLUME: %d%%"), v); + mainwin_lock_info_text(volume_msg); + g_free(volume_msg); + + if (balance < 0) + input_set_volume(v, (v * (100 - abs(balance))) / 100); + else if (balance > 0) + input_set_volume((v * (100 - abs(balance))) / 100, v); + else + input_set_volume(v, v); +} + +void +mainwin_adjust_volume_release(void) +{ + mainwin_release_info_text(); + setting_volume = FALSE; + read_volume(VOLUME_ADJUSTED); +} + +void +mainwin_adjust_balance_motion(gint b) +{ + gchar *balance_msg; + gint v, pvl, pvr; + + setting_volume = TRUE; + balance = b; + input_get_volume(&pvl, &pvr); + v = MAX(pvl, pvr); + if (b < 0) { + balance_msg = g_strdup_printf(_("BALANCE: %d%% LEFT"), -b); + input_set_volume(v, (gint) rint(((100 + b) / 100.0) * v)); + } + else if (b == 0) { + balance_msg = g_strdup_printf(_("BALANCE: CENTER")); + input_set_volume(v, v); + } + else { /* b > 0 */ + balance_msg = g_strdup_printf(_("BALANCE: %d%% RIGHT"), b); + input_set_volume((gint) rint(((100 - b) / 100.0) * v), v); + } + mainwin_lock_info_text(balance_msg); + g_free(balance_msg); +} + +void +mainwin_adjust_balance_release(void) +{ + mainwin_release_info_text(); + setting_volume = FALSE; + read_volume(VOLUME_ADJUSTED); +} + +void +mainwin_set_volume_slider(gint percent) +{ + hslider_set_position(mainwin_volume, (gint) rint((percent * 51) / 100.0)); +} + +void +mainwin_set_balance_slider(gint percent) +{ + hslider_set_position(mainwin_balance, + (gint) rint(((percent * 12) / 100.0) + 12)); +} + +void +mainwin_volume_motion_cb(gint pos) +{ + gint vol = (pos * 100) / 51; + mainwin_adjust_volume_motion(vol); + equalizerwin_set_volume_slider(vol); +} + +void +mainwin_volume_release_cb(gint pos) +{ + mainwin_adjust_volume_release(); +} + +gint +mainwin_balance_frame_cb(gint pos) +{ + return ((abs(pos - 12) * 28) / 13); +} + +void +mainwin_balance_motion_cb(gint pos) +{ + gint bal = ((pos - 12) * 100) / 12; + mainwin_adjust_balance_motion(bal); + equalizerwin_set_balance_slider(bal); +} + +void +mainwin_balance_release_cb(gint pos) +{ + mainwin_adjust_volume_release(); +} + +void +mainwin_set_volume_diff(gint diff) +{ + gint vl, vr, vol; + + input_get_volume(&vl, &vr); + vol = MAX(vl, vr); + vol = CLAMP(vol + diff, 0, 100); + + mainwin_adjust_volume_motion(vol); + setting_volume = FALSE; + mainwin_set_volume_slider(vol); + equalizerwin_set_volume_slider(vol); + read_volume(VOLUME_SET); +} + +void +mainwin_set_balance_diff(gint diff) +{ + gint b; + b = CLAMP(balance + diff, -100, 100); + mainwin_adjust_balance_motion(b); + setting_volume = FALSE; + mainwin_set_balance_slider(b); + equalizerwin_set_balance_slider(b); + read_volume(VOLUME_SET); +} + +void +mainwin_show(gboolean show) +{ + if (show) + mainwin_real_show(); + else + mainwin_real_hide(); +} + +void +mainwin_real_show(void) +{ + cfg.player_visible = TRUE; + + check_set( toggleaction_group_others , "show player" , TRUE ); + + if (cfg.player_shaded) + vis_clear_data(active_vis); + + mainwin_vis_set_active_vis(MAINWIN_VIS_ACTIVE_MAINWIN); + mainwin_set_shape_mask(); + + if (cfg.show_wm_decorations) { + if (!pposition_broken && cfg.player_x != -1 + && cfg.save_window_position) + gtk_window_move(GTK_WINDOW(mainwin), cfg.player_x, cfg.player_y); + + gtk_widget_show(mainwin); + + if (pposition_broken && cfg.player_x != -1 + && cfg.save_window_position) + gtk_window_move(GTK_WINDOW(mainwin), cfg.player_x, cfg.player_y); + + return; + } + + gtk_widget_show_all(mainwin); + + if (!nullmask) + return; + + g_object_unref(nullmask); + nullmask = NULL; + + gdk_window_set_hints(mainwin->window, 0, 0, + !bmp_active_skin->properties.mainwin_width ? PLAYER_WIDTH : + bmp_active_skin->properties.mainwin_width, + !bmp_active_skin->properties.mainwin_height ? PLAYER_HEIGHT : + bmp_active_skin->properties.mainwin_height, + !bmp_active_skin->properties.mainwin_width ? PLAYER_WIDTH : + bmp_active_skin->properties.mainwin_width, + !bmp_active_skin->properties.mainwin_height ? PLAYER_HEIGHT : + bmp_active_skin->properties.mainwin_height, + GDK_HINT_MIN_SIZE | GDK_HINT_MAX_SIZE); + gtk_window_resize(GTK_WINDOW(mainwin), + !bmp_active_skin->properties.mainwin_width ? PLAYER_WIDTH : + bmp_active_skin->properties.mainwin_width, + !bmp_active_skin->properties.mainwin_height ? PLAYER_HEIGHT : + bmp_active_skin->properties.mainwin_height); + if (cfg.player_x != -1 && cfg.player_y != -1) + gtk_window_move(GTK_WINDOW(mainwin), cfg.player_x, cfg.player_y); + + draw_main_window(TRUE); + + gtk_window_present(GTK_WINDOW(mainwin)); +} + +void +mainwin_real_hide(void) +{ + GdkGC *gc; + GdkColor pattern; + + check_set( toggleaction_group_others , "show player", FALSE); + + if (cfg.player_shaded) + svis_clear_data(mainwin_svis); + + if (!cfg.show_wm_decorations) { + nullmask = gdk_pixmap_new(mainwin->window, 20, 20, 1); + gc = gdk_gc_new(nullmask); + pattern.pixel = 0; + gdk_gc_set_foreground(gc, &pattern); + gdk_draw_rectangle(nullmask, gc, TRUE, 0, 0, 20, 20); + g_object_unref(gc); + gtk_widget_shape_combine_mask(mainwin, nullmask, 0, 0); + + gdk_window_set_hints(mainwin->window, 0, 0, 0, 0, 0, 0, + GDK_HINT_MIN_SIZE | GDK_HINT_MAX_SIZE); + gdk_window_resize(mainwin->window, 0, 0); + } + + gtk_widget_hide(mainwin); + + mainwin_vis_set_active_vis(MAINWIN_VIS_ACTIVE_PLAYLISTWIN); + cfg.player_visible = FALSE; +} + + +void +mainwin_set_stopaftersong(gboolean stop) +{ + cfg.stopaftersong = stop; + check_set(toggleaction_group_others, "stop after current song", cfg.stopaftersong); +} + + +static void +mainwin_set_doublesize(gboolean doublesize) +{ + gint height; + + if (cfg.player_shaded) + height = MAINWIN_SHADED_HEIGHT; + else + height = bmp_active_skin->properties.mainwin_height; + + mainwin_set_shape_mask(); + + dock_window_resize(GTK_WINDOW(mainwin), cfg.player_shaded ? MAINWIN_SHADED_WIDTH : bmp_active_skin->properties.mainwin_width, + cfg.player_shaded ? MAINWIN_SHADED_HEIGHT : bmp_active_skin->properties.mainwin_height, + bmp_active_skin->properties.mainwin_width * 2, bmp_active_skin->properties.mainwin_height * 2); + + if (cfg.doublesize) { + gdk_window_set_back_pixmap(mainwin->window, mainwin_bg_x2, 0); + } + else { + gdk_window_set_back_pixmap(mainwin->window, mainwin_bg, 0); + } + + draw_main_window(TRUE); + vis_set_doublesize(mainwin_vis, doublesize); +} + +void +set_doublesize(gboolean doublesize) +{ + cfg.doublesize = doublesize; + + mainwin_set_doublesize(doublesize); + + if (cfg.eq_doublesize_linked) + equalizerwin_set_doublesize(doublesize); +} + + + +void +mainwin_general_menu_callback(gpointer data, + guint action, + GtkWidget * item) +{ + Playlist *playlist = playlist_get_active(); + + switch (action) { + case MAINWIN_GENERAL_PREFS: + show_prefs_window(); + break; + case MAINWIN_GENERAL_ABOUT: + show_about_window(); + break; + case MAINWIN_GENERAL_PLAYFILE: + util_run_filebrowser(NO_PLAY_BUTTON); + break; + case MAINWIN_GENERAL_PLAYCD: + play_medium(); + break; + case MAINWIN_GENERAL_ADDCD: + add_medium(); + break; + case MAINWIN_GENERAL_PLAYLOCATION: + mainwin_show_add_url_window(); + break; + case MAINWIN_GENERAL_FILEINFO: + playlist_fileinfo_current(playlist); + break; + case MAINWIN_GENERAL_FOCUSPLWIN: + gtk_window_present(GTK_WINDOW(playlistwin)); + break; + case MAINWIN_GENERAL_SHOWMWIN: + mainwin_show(GTK_CHECK_MENU_ITEM(item)->active); + break; + case MAINWIN_GENERAL_SHOWPLWIN: + if (GTK_CHECK_MENU_ITEM(item)->active) + playlistwin_show(); + else + playlistwin_hide(); + break; + case MAINWIN_GENERAL_SHOWEQWIN: + if (GTK_CHECK_MENU_ITEM(item)->active) + equalizerwin_real_show(); + else + equalizerwin_real_hide(); + break; + case MAINWIN_GENERAL_PREV: + playlist_prev(playlist); + break; + case MAINWIN_GENERAL_PLAY: + mainwin_play_pushed(); + break; + case MAINWIN_GENERAL_PAUSE: + playback_pause(); + break; + case MAINWIN_GENERAL_STOP: + mainwin_stop_pushed(); + break; + case MAINWIN_GENERAL_NEXT: + playlist_next(playlist); + break; + case MAINWIN_GENERAL_BACK5SEC: + if (playback_get_playing() + && playlist_get_current_length(playlist) != -1) + playback_seek_relative(-5); + break; + case MAINWIN_GENERAL_FWD5SEC: + if (playback_get_playing() + && playlist_get_current_length(playlist) != -1) + playback_seek_relative(5); + break; + case MAINWIN_GENERAL_START: + playlist_set_position(playlist, 0); + break; + case MAINWIN_GENERAL_JTT: + mainwin_jump_to_time(); + break; + case MAINWIN_GENERAL_JTF: + mainwin_jump_to_file(); + break; + case MAINWIN_GENERAL_EXIT: + mainwin_quit_cb(); + break; + case MAINWIN_GENERAL_SETAB: + if (playlist_get_current_length(playlist) != -1) { + if (ab_position_a == -1) { + ab_position_a = playback_get_time(); + ab_position_b = -1; + mainwin_lock_info_text("LOOP-POINT A POSITION SET."); + } else if (ab_position_b == -1) { + int time = playback_get_time(); + if (time > ab_position_a) + ab_position_b = time; + mainwin_release_info_text(); + } else { + ab_position_a = playback_get_time(); + ab_position_b = -1; + mainwin_lock_info_text("LOOP-POINT A POSITION RESET."); + } + } + break; + case MAINWIN_GENERAL_CLEARAB: + if (playlist_get_current_length(playlist) != -1) { + ab_position_a = ab_position_b = -1; + mainwin_release_info_text(); + } + break; + case MAINWIN_GENERAL_NEW_PL: + { + Playlist *new_pl = playlist_new(); + playlist_add_playlist(new_pl); + playlist_select_playlist(new_pl); + } + break; + case MAINWIN_GENERAL_PREV_PL: + playlist_select_prev(); + break; + case MAINWIN_GENERAL_NEXT_PL: + playlist_select_next(); + break; + } +} + +static void +mainwin_mr_change(MenuRowItem i) +{ + switch (i) { + case MENUROW_NONE: + mainwin_set_info_text(); + break; + case MENUROW_OPTIONS: + mainwin_lock_info_text(_("OPTIONS MENU")); + break; + case MENUROW_ALWAYS: + if (mainwin_menurow->mr_always_selected) + mainwin_lock_info_text(_("DISABLE ALWAYS ON TOP")); + else + mainwin_lock_info_text(_("ENABLE ALWAYS ON TOP")); + break; + case MENUROW_FILEINFOBOX: + mainwin_lock_info_text(_("FILE INFO BOX")); + break; + case MENUROW_DOUBLESIZE: + if (mainwin_menurow->mr_doublesize_selected) + mainwin_lock_info_text(_("DISABLE DOUBLESIZE")); + else + mainwin_lock_info_text(_("ENABLE DOUBLESIZE")); + break; + case MENUROW_VISUALIZATION: + mainwin_lock_info_text(_("VISUALIZATION MENU")); + break; + } +} + +static void +mainwin_mr_release(MenuRowItem i) +{ + GdkModifierType modmask; + GtkWidget *widget; + gint x, y; + + switch (i) { + case MENUROW_OPTIONS: + gdk_window_get_pointer(NULL, &x, &y, &modmask); + ui_manager_popup_menu_show(GTK_MENU(mainwin_view_menu), x, y, 1, + GDK_CURRENT_TIME); + break; + case MENUROW_ALWAYS: + gtk_toggle_action_set_active( + GTK_TOGGLE_ACTION(gtk_action_group_get_action( + toggleaction_group_others , "view always on top" )) , + mainwin_menurow->mr_always_selected ); + break; + case MENUROW_FILEINFOBOX: + playlist_fileinfo_current(playlist_get_active()); + break; + case MENUROW_DOUBLESIZE: + gtk_toggle_action_set_active( + GTK_TOGGLE_ACTION(gtk_action_group_get_action( + toggleaction_group_others , "view doublesize" )) , + mainwin_menurow->mr_doublesize_selected ); + break; + case MENUROW_VISUALIZATION: + gdk_window_get_pointer(NULL, &x, &y, &modmask); + ui_manager_popup_menu_show(GTK_MENU(mainwin_visualization_menu), x, y, 1, GDK_CURRENT_TIME); + break; + case MENUROW_NONE: + break; + } + mainwin_release_info_text(); +} + +static void +run_no_audiocd_dialog(void) +{ + const gchar *markup = + N_("<b><big>No playable CD found.</big></b>\n\n" + "No CD inserted, or inserted CD is not an audio CD.\n"); + + GtkWidget *dialog = + gtk_message_dialog_new_with_markup(GTK_WINDOW(mainwin), + GTK_DIALOG_DESTROY_WITH_PARENT, + GTK_MESSAGE_ERROR, + GTK_BUTTONS_OK, + _(markup)); + gtk_dialog_run(GTK_DIALOG(dialog)); + gtk_widget_destroy(dialog); +} + +static void +run_no_output_device_dialog(void) +{ + const gchar *markup = + N_("<b><big>Couldn't open audio.</big></b>\n\n" + "Please check that:\n" + "1. You have the correct output plugin selected.\n" + "2. No other programs is blocking the soundcard.\n" + "3. Your soundcard is configured properly.\n"); + + GtkWidget *dialog = + gtk_message_dialog_new_with_markup(GTK_WINDOW(mainwin), + GTK_DIALOG_DESTROY_WITH_PARENT, + GTK_MESSAGE_ERROR, + GTK_BUTTONS_OK, + _(markup)); + gtk_dialog_run(GTK_DIALOG(dialog)); + gtk_widget_destroy(dialog); +} + + +void +add_medium(void) +{ + GList *list, *node; + gchar *filename; + gchar *path; + ConfigDb *db; + + db = bmp_cfg_db_open(); + + bmp_cfg_db_get_string(db, "CDDA", "directory", &path); + bmp_cfg_db_close(db); + + if (!(list = input_scan_dir(path))) { + run_no_audiocd_dialog(); + return; + } + + for (node = list; node; node = g_list_next(node)) { + filename = g_build_filename(path, node->data, NULL); + playlist_add(playlist_get_active(), filename); + g_free(filename); + g_free(node->data); + } + + g_free(path); + g_list_free(list); + +} + +void +play_medium(void) +{ + GList *list, *node; + gchar *filename; + gchar *path; + ConfigDb *db; + Playlist *playlist = playlist_get_active(); + + db = bmp_cfg_db_open(); + bmp_cfg_db_get_string(db, "CDDA", "directory", &path); + bmp_cfg_db_close(db); + + if (!(list = input_scan_dir(path))) { + run_no_audiocd_dialog(); + return; + } + + playlist_clear(playlist); + + for (node = list; node; node = g_list_next(node)) { + filename = g_build_filename(path, node->data, NULL); + playlist_add(playlist, filename); + g_free(filename); + g_free(node->data); + } + + g_free(path); + g_list_free(list); + + playlist_set_position(playlist, 0); + playback_initiate(); +} + +void +read_volume(gint when) +{ + static gint pvl = 0, pvr = 0; + static gint times = VOLSET_DISP_TIMES; + static gboolean changing = FALSE; + + gint vl, vr, b, v; + + input_get_volume(&vl, &vr); + + switch (when) { + case VOLSET_STARTUP: + vl = CLAMP(vl, 0, 100); + vr = CLAMP(vr, 0, 100); + pvl = vl; + pvr = vr; + v = MAX(vl, vr); + if (vl > vr) + b = (gint) rint(((gdouble) vr / vl) * 100) - 100; + else if (vl < vr) + b = 100 - (gint) rint(((gdouble) vl / vr) * 100); + else + b = 0; + + balance = b; + mainwin_set_volume_slider(v); + equalizerwin_set_volume_slider(v); + mainwin_set_balance_slider(b); + equalizerwin_set_balance_slider(b); + return; + + case VOLSET_UPDATE: + if (vl == -1 || vr == -1) + return; + + if (setting_volume) { + pvl = vl; + pvr = vr; + return; + } + + if (pvr == vr && pvl == vl && changing) { + if (times < VOLSET_DISP_TIMES) + times++; + else { + mainwin_release_info_text(); + changing = FALSE; + } + } + else if (pvr != vr || pvl != vl) { + gchar *tmp; + + v = MAX(vl, vr); + if (vl > vr) + b = (gint) rint(((gdouble) vr / vl) * 100) - 100; + else if (vl < vr) + b = 100 - (gint) rint(((gdouble) vl / vr) * 100); + else + b = 0; + + if (MAX(vl, vr) != MAX(pvl, pvr)) + tmp = g_strdup_printf(_("VOLUME: %d%%"), v); + else { + if (vl > vr) { + tmp = g_strdup_printf(_("BALANCE: %d%% LEFT"), -b); + } + else if (vr == vl) + tmp = g_strdup_printf(_("BALANCE: CENTER")); + else { /* (vl < vr) */ + tmp = g_strdup_printf(_("BALANCE: %d%% RIGHT"), b); + } + } + mainwin_lock_info_text(tmp); + g_free(tmp); + + pvr = vr; + pvl = vl; + times = 0; + changing = TRUE; + mainwin_set_volume_slider(v); + equalizerwin_set_volume_slider(v); + + /* Don't change the balance slider if the volume has been + * set to zero. The balance can be anything, and our best + * guess is what is was before. */ + if (v > 0) { + balance = b; + mainwin_set_balance_slider(b); + equalizerwin_set_balance_slider(b); + } + } + break; + + case VOLUME_ADJUSTED: + pvl = vl; + pvr = vr; + break; + + case VOLUME_SET: + times = 0; + changing = TRUE; + pvl = vl; + pvr = vr; + break; + } +} + + +/* TODO: HAL! */ +gboolean +can_play_cd(void) +{ + GList *ilist; + + for (ilist = get_input_list(); ilist; ilist = g_list_next(ilist)) { + InputPlugin *ip = INPUT_PLUGIN(ilist->data); + + if (!g_ascii_strcasecmp(g_basename(ip->filename), + PLUGIN_FILENAME("cdaudio"))) { + return TRUE; + } + } + + return FALSE; +} + + +static void +set_timer_mode(TimerMode mode) +{ + if (mode == TIMER_ELAPSED) + check_set(radioaction_group_viewtime, "view time elapsed", TRUE); + else + check_set(radioaction_group_viewtime, "view time remaining", TRUE); +} + +static void +set_timer_mode_menu_cb(TimerMode mode) +{ + cfg.timer_mode = mode; +} + +static void +mainwin_playlist_prev(void) +{ + playlist_prev(playlist_get_active()); +} + +static void +mainwin_playlist_next(void) +{ + playlist_next(playlist_get_active()); +} + +void +mainwin_setup_menus(void) +{ + set_timer_mode(cfg.timer_mode); + + /* View menu */ + + check_set(toggleaction_group_others, "view always on top", cfg.always_on_top); + check_set(toggleaction_group_others, "view put on all workspaces", cfg.sticky); + check_set(toggleaction_group_others, "roll up player", cfg.player_shaded); + check_set(toggleaction_group_others, "roll up playlist editor", cfg.playlist_shaded); + check_set(toggleaction_group_others, "roll up equalizer", cfg.equalizer_shaded); + check_set(toggleaction_group_others, "view easy move", cfg.easy_move); + check_set(toggleaction_group_others, "view doublesize", cfg.doublesize); + + /* Songname menu */ + + check_set(toggleaction_group_others, "autoscroll songname", cfg.autoscroll); + check_set(toggleaction_group_others, "stop after current song", cfg.stopaftersong); + + /* Playback menu */ + + check_set(toggleaction_group_others, "playback repeat", cfg.repeat); + check_set(toggleaction_group_others, "playback shuffle", cfg.shuffle); + check_set(toggleaction_group_others, "playback no playlist advance", cfg.no_playlist_advance); + + /* Visualization menu */ + + switch ( cfg.vis_type ) + { + case VIS_ANALYZER: + check_set(radioaction_group_vismode, "vismode analyzer", TRUE); + break; + case VIS_SCOPE: + check_set(radioaction_group_vismode, "vismode scope", TRUE); + break; + case VIS_VOICEPRINT: + check_set(radioaction_group_vismode, "vismode voiceprint", TRUE); + break; + case VIS_OFF: + default: + check_set(radioaction_group_vismode, "vismode off", TRUE); + break; + } + + switch ( cfg.analyzer_mode ) + { + case ANALYZER_FIRE: + check_set(radioaction_group_anamode, "anamode fire", TRUE); + break; + case ANALYZER_VLINES: + check_set(radioaction_group_anamode, "anamode vertical lines", TRUE); + break; + case ANALYZER_NORMAL: + default: + check_set(radioaction_group_anamode, "anamode normal", TRUE); + break; + } + + switch ( cfg.analyzer_type ) + { + case ANALYZER_BARS: + check_set(radioaction_group_anatype, "anatype bars", TRUE); + break; + case ANALYZER_LINES: + default: + check_set(radioaction_group_anatype, "anatype lines", TRUE); + break; + } + + check_set(toggleaction_group_others, "anamode peaks", cfg.analyzer_peaks ); + + switch ( cfg.scope_mode ) + { + case SCOPE_LINE: + check_set(radioaction_group_scomode, "scomode line", TRUE); + break; + case SCOPE_SOLID: + check_set(radioaction_group_scomode, "scomode solid", TRUE); + break; + case SCOPE_DOT: + default: + check_set(radioaction_group_scomode, "scomode dot", TRUE); + break; + } + + switch ( cfg.voiceprint_mode ) + { + case VOICEPRINT_FIRE: + check_set(radioaction_group_vprmode, "vprmode fire", TRUE); + break; + case VOICEPRINT_ICE: + check_set(radioaction_group_vprmode, "vprmode ice", TRUE); + break; + case VOICEPRINT_NORMAL: + default: + check_set(radioaction_group_vprmode, "vprmode normal", TRUE); + break; + } + + switch ( cfg.vu_mode ) + { + case VU_SMOOTH: + check_set(radioaction_group_wshmode, "wshmode smooth", TRUE); + break; + case VU_NORMAL: + default: + check_set(radioaction_group_wshmode, "wshmode normal", TRUE); + break; + } + + switch ( cfg.vis_refresh ) + { + case REFRESH_HALF: + check_set(radioaction_group_refrate, "refrate half", TRUE); + break; + case REFRESH_QUARTER: + check_set(radioaction_group_refrate, "refrate quarter", TRUE); + break; + case REFRESH_EIGTH: + check_set(radioaction_group_refrate, "refrate eighth", TRUE); + break; + case REFRESH_FULL: + default: + check_set(radioaction_group_refrate, "refrate full", TRUE); + break; + } + + switch ( cfg.analyzer_falloff ) + { + case FALLOFF_SLOW: + check_set(radioaction_group_anafoff, "anafoff slow", TRUE); + break; + case FALLOFF_MEDIUM: + check_set(radioaction_group_anafoff, "anafoff medium", TRUE); + break; + case FALLOFF_FAST: + check_set(radioaction_group_anafoff, "anafoff fast", TRUE); + break; + case FALLOFF_FASTEST: + check_set(radioaction_group_anafoff, "anafoff fastest", TRUE); + break; + case FALLOFF_SLOWEST: + default: + check_set(radioaction_group_anafoff, "anafoff slowest", TRUE); + break; + } + + switch ( cfg.peaks_falloff ) + { + case FALLOFF_SLOW: + check_set(radioaction_group_peafoff, "peafoff slow", TRUE); + break; + case FALLOFF_MEDIUM: + check_set(radioaction_group_peafoff, "peafoff medium", TRUE); + break; + case FALLOFF_FAST: + check_set(radioaction_group_peafoff, "peafoff fast", TRUE); + break; + case FALLOFF_FASTEST: + check_set(radioaction_group_peafoff, "peafoff fastest", TRUE); + break; + case FALLOFF_SLOWEST: + default: + check_set(radioaction_group_peafoff, "peafoff slowest", TRUE); + break; + } + +} + +static void +mainwin_create_widgets(void) +{ + mainwin_menubtn = + create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 6, 3, 9, 9, + 0, 0, 0, 9, mainwin_menubtn_cb, SKIN_TITLEBAR); + mainwin_menubtn->pb_allow_draw = FALSE; + mainwin_minimize = + create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 244, 3, 9, + 9, 9, 0, 9, 9, mainwin_minimize_cb, SKIN_TITLEBAR); + mainwin_minimize->pb_allow_draw = FALSE; + mainwin_shade = + create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 254, 3, 9, + 9, 0, cfg.player_shaded ? 27 : 18, 9, + cfg.player_shaded ? 27 : 18, mainwin_shade_toggle, + SKIN_TITLEBAR); + mainwin_shade->pb_allow_draw = FALSE; + mainwin_close = + create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 264, 3, 9, + 9, 18, 0, 18, 9, mainwin_quit_cb, SKIN_TITLEBAR); + mainwin_close->pb_allow_draw = FALSE; + + mainwin_rew = + create_pbutton_ex(&mainwin_wlist, mainwin_bg, mainwin_gc, 16, 88, 23, + 18, 0, 0, 0, 18, mainwin_rev_pushed, mainwin_rev_release, + SKIN_CBUTTONS, SKIN_CBUTTONS); + mainwin_play = + create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 39, 88, 23, + 18, 23, 0, 23, 18, mainwin_play_pushed, SKIN_CBUTTONS); + mainwin_pause = + create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 62, 88, 23, + 18, 46, 0, 46, 18, playback_pause, SKIN_CBUTTONS); + mainwin_stop = + create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 85, 88, 23, + 18, 69, 0, 69, 18, mainwin_stop_pushed, SKIN_CBUTTONS); +#if 0 + mainwin_fwd = + create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 108, 88, 22, + 18, 92, 0, 92, 18, playlist_next, SKIN_CBUTTONS); +#endif + mainwin_fwd = + create_pbutton_ex(&mainwin_wlist, mainwin_bg, mainwin_gc, 108, 88, 22, + 18, 92, 0, 92, 18, mainwin_fwd_pushed, mainwin_fwd_release, + SKIN_CBUTTONS, SKIN_CBUTTONS); + + mainwin_eject = + create_pbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 136, 89, 22, + 16, 114, 0, 114, 16, mainwin_eject_pushed, + SKIN_CBUTTONS); + + mainwin_srew = + create_sbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 169, 4, 8, + 7, mainwin_playlist_prev); + mainwin_splay = + create_sbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 177, 4, 10, + 7, mainwin_play_pushed); + mainwin_spause = + create_sbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 187, 4, 10, + 7, playback_pause); + mainwin_sstop = + create_sbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 197, 4, 9, + 7, mainwin_stop_pushed); + mainwin_sfwd = + create_sbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 206, 4, 8, + 7, mainwin_playlist_next); + mainwin_seject = + create_sbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 216, 4, 9, + 7, mainwin_eject_pushed); + + mainwin_shuffle = + create_tbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 164, 89, 46, + 15, 28, 0, 28, 15, 28, 30, 28, 45, + mainwin_shuffle_pushed, SKIN_SHUFREP); + + mainwin_repeat = + create_tbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 210, 89, 28, + 15, 0, 0, 0, 15, 0, 30, 0, 45, + mainwin_repeat_pushed, SKIN_SHUFREP); + + mainwin_eq = + create_tbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 219, 58, 23, + 12, 0, 61, 46, 61, 0, 73, 46, 73, equalizerwin_show, + SKIN_SHUFREP); + tbutton_set_toggled(mainwin_eq, cfg.equalizer_visible); + mainwin_pl = + create_tbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 242, 58, 23, + 12, 23, 61, 69, 61, 23, 73, 69, 73, + mainwin_pl_pushed, SKIN_SHUFREP); + tbutton_set_toggled(mainwin_pl, cfg.playlist_visible); + + mainwin_info = + create_textbox(&mainwin_wlist, mainwin_bg, mainwin_gc, 112, 27, + 153, 1, SKIN_TEXT); + textbox_set_scroll(mainwin_info, cfg.autoscroll); + textbox_set_xfont(mainwin_info, cfg.mainwin_use_xfont, cfg.mainwin_font); + + mainwin_othertext = + create_textbox(&mainwin_wlist, mainwin_bg, mainwin_gc, 112, 43, + 153, 1, SKIN_TEXT); + + mainwin_rate_text = + create_textbox(&mainwin_wlist, mainwin_bg, mainwin_gc, 111, 43, 15, + 0, SKIN_TEXT); + mainwin_freq_text = + create_textbox(&mainwin_wlist, mainwin_bg, mainwin_gc, 156, 43, 10, + 0, SKIN_TEXT); + + mainwin_menurow = + create_menurow(&mainwin_wlist, mainwin_bg, mainwin_gc, 10, 22, 304, + 0, 304, 44, mainwin_mr_change, mainwin_mr_release, + SKIN_TITLEBAR); + mainwin_menurow->mr_doublesize_selected = cfg.doublesize; + mainwin_menurow->mr_always_selected = cfg.always_on_top; + + mainwin_volume = + create_hslider(&mainwin_wlist, mainwin_bg, mainwin_gc, 107, 57, 68, + 13, 15, 422, 0, 422, 14, 11, 15, 0, 0, 51, + mainwin_volume_frame_cb, mainwin_volume_motion_cb, + mainwin_volume_release_cb, SKIN_VOLUME); + mainwin_balance = + create_hslider(&mainwin_wlist, mainwin_bg, mainwin_gc, 177, 57, 38, + 13, 15, 422, 0, 422, 14, 11, 15, 9, 0, 24, + mainwin_balance_frame_cb, mainwin_balance_motion_cb, + mainwin_balance_release_cb, SKIN_BALANCE); + + mainwin_monostereo = + create_monostereo(&mainwin_wlist, mainwin_bg, mainwin_gc, 212, 41, + SKIN_MONOSTEREO); + + mainwin_playstatus = + create_playstatus(&mainwin_wlist, mainwin_bg, mainwin_gc, 24, 28); + + mainwin_minus_num = + create_number(&mainwin_wlist, mainwin_bg, mainwin_gc, 36, 26, + SKIN_NUMBERS); + widget_hide(WIDGET(mainwin_minus_num)); + mainwin_10min_num = + create_number(&mainwin_wlist, mainwin_bg, mainwin_gc, 48, 26, + SKIN_NUMBERS); + widget_hide(WIDGET(mainwin_10min_num)); + + mainwin_min_num = + create_number(&mainwin_wlist, mainwin_bg, mainwin_gc, 60, 26, + SKIN_NUMBERS); + widget_hide(WIDGET(mainwin_min_num)); + + mainwin_10sec_num = + create_number(&mainwin_wlist, mainwin_bg, mainwin_gc, 78, 26, + SKIN_NUMBERS); + widget_hide(WIDGET(mainwin_10sec_num)); + + mainwin_sec_num = + create_number(&mainwin_wlist, mainwin_bg, mainwin_gc, 90, 26, + SKIN_NUMBERS); + widget_hide(WIDGET(mainwin_sec_num)); + + mainwin_about = + create_sbutton(&mainwin_wlist, mainwin_bg, mainwin_gc, 247, 83, 20, + 25, show_about_window); + + mainwin_vis = + create_vis(&mainwin_wlist, mainwin_bg, mainwin->window, mainwin_gc, + 24, 43, 76, cfg.doublesize); + mainwin_svis = create_svis(&mainwin_wlist, mainwin_bg, mainwin_gc, 79, 5); + active_vis = mainwin_vis; + + mainwin_position = + create_hslider(&mainwin_wlist, mainwin_bg, mainwin_gc, 16, 72, 248, + 10, 248, 0, 278, 0, 29, 10, 10, 0, 0, 219, NULL, + mainwin_position_motion_cb, + mainwin_position_release_cb, SKIN_POSBAR); + widget_hide(WIDGET(mainwin_position)); + + mainwin_sposition = + create_hslider(&mainwin_wlist, mainwin_bg, mainwin_gc, 226, 4, 17, + 7, 17, 36, 17, 36, 3, 7, 36, 0, 1, 13, + mainwin_spos_frame_cb, mainwin_spos_motion_cb, + mainwin_spos_release_cb, SKIN_TITLEBAR); + widget_hide(WIDGET(mainwin_sposition)); + + mainwin_stime_min = + create_textbox(&mainwin_wlist, mainwin_bg, mainwin_gc, 130, 4, 15, + FALSE, SKIN_TEXT); + mainwin_stime_sec = + create_textbox(&mainwin_wlist, mainwin_bg, mainwin_gc, 147, 4, 10, + FALSE, SKIN_TEXT); + + if (!cfg.player_shaded) { + widget_hide(WIDGET(mainwin_svis)); + widget_hide(WIDGET(mainwin_srew)); + widget_hide(WIDGET(mainwin_splay)); + widget_hide(WIDGET(mainwin_spause)); + widget_hide(WIDGET(mainwin_sstop)); + widget_hide(WIDGET(mainwin_sfwd)); + widget_hide(WIDGET(mainwin_seject)); + widget_hide(WIDGET(mainwin_stime_min)); + widget_hide(WIDGET(mainwin_stime_sec)); + } + + err = gtk_message_dialog_new(GTK_WINDOW(mainwin), GTK_DIALOG_DESTROY_WITH_PARENT|GTK_DIALOG_MODAL, + GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, _("Error in Audacious.")); + + + gtk_window_set_position(GTK_WINDOW(err), GTK_WIN_POS_CENTER); + /* Dang well better set an error message or you'll see this */ + gtk_message_dialog_format_secondary_text(GTK_MESSAGE_DIALOG(err), + "Boo! Bad stuff! Booga Booga!"); + +} + +static void +mainwin_create_window(void) +{ + gint width, height; + + mainwin = gtk_window_new(GTK_WINDOW_TOPLEVEL); + gtk_window_set_title(GTK_WINDOW(mainwin), _("Audacious")); + gtk_window_set_role(GTK_WINDOW(mainwin), "player"); + gtk_window_set_resizable(GTK_WINDOW(mainwin), FALSE); + + width = cfg.player_shaded ? MAINWIN_SHADED_WIDTH : bmp_active_skin->properties.mainwin_width; + height = cfg.player_shaded ? MAINWIN_SHADED_HEIGHT : bmp_active_skin->properties.mainwin_height; + + if (cfg.doublesize) { + width *= 2; + height *= 2; + } + + gtk_widget_set_size_request(mainwin, width, height); + gtk_widget_set_app_paintable(mainwin, TRUE); + + dock_window_list = dock_window_set_decorated(dock_window_list, + GTK_WINDOW(mainwin), + cfg.show_wm_decorations); + + if (cfg.player_x != -1 && cfg.save_window_position) + gtk_window_move(GTK_WINDOW(mainwin), cfg.player_x, cfg.player_y); + + gtk_widget_add_events(mainwin, + GDK_FOCUS_CHANGE_MASK | GDK_BUTTON_MOTION_MASK | + GDK_BUTTON_PRESS_MASK | GDK_BUTTON_RELEASE_MASK | + GDK_SCROLL_MASK | GDK_KEY_PRESS_MASK | + GDK_VISIBILITY_NOTIFY_MASK); + gtk_widget_realize(mainwin); + + util_set_cursor(mainwin); + + g_signal_connect(mainwin, "destroy", G_CALLBACK(mainwin_destroy), NULL); + g_signal_connect(mainwin, "button_press_event", + G_CALLBACK(mainwin_mouse_button_press), NULL); + g_signal_connect(mainwin, "scroll_event", + G_CALLBACK(mainwin_scrolled), NULL); + g_signal_connect(mainwin, "button_release_event", + G_CALLBACK(mainwin_mouse_button_release), NULL); + g_signal_connect(mainwin, "motion_notify_event", + G_CALLBACK(mainwin_motion), NULL); + g_signal_connect_after(mainwin, "focus_in_event", + G_CALLBACK(mainwin_focus_in), NULL); + g_signal_connect_after(mainwin, "focus_out_event", + G_CALLBACK(mainwin_focus_out), NULL); + g_signal_connect(mainwin, "configure_event", + G_CALLBACK(mainwin_configure), NULL); + g_signal_connect(mainwin, "style_set", + G_CALLBACK(mainwin_set_back_pixmap), NULL); + + bmp_drag_dest_set(mainwin); + + g_signal_connect(mainwin, "key_press_event", + G_CALLBACK(mainwin_keypress), NULL); +} + +void +mainwin_create(void) +{ + mainwin_create_window(); + + gtk_window_add_accel_group( GTK_WINDOW(mainwin) , ui_manager_get_accel_group() ); + + mainwin_gc = gdk_gc_new(mainwin->window); + mainwin_bg = gdk_pixmap_new(mainwin->window, + bmp_active_skin->properties.mainwin_width, + bmp_active_skin->properties.mainwin_height, -1); + mainwin_bg_x2 = gdk_pixmap_new(mainwin->window, + bmp_active_skin->properties.mainwin_width * 2, + bmp_active_skin->properties.mainwin_height * 2, -1); + mainwin_set_back_pixmap(); + mainwin_create_widgets(); + + vis_set_window(mainwin_vis, mainwin->window); +} + +void +mainwin_attach_idle_func(void) +{ + mainwin_timeout_id = g_timeout_add(MAINWIN_UPDATE_INTERVAL, + mainwin_idle_func, NULL); +} + +static void +idle_func_update_song_info(gint time) +{ + gint length, t; + gchar stime_prefix; + + if (ab_position_a != -1 && ab_position_b != -1 && time > ab_position_b) + playback_seek(ab_position_a/1000); + + length = playlist_get_current_length(playlist_get_active()); + if (playback_get_playing()) + playlistwin_set_time(time, length, cfg.timer_mode); + else + playlistwin_hide_timer(); + input_update_vis(time); + + if (cfg.timer_mode == TIMER_REMAINING) { + if (length != -1) { + number_set_number(mainwin_minus_num, 11); + t = length - time; + stime_prefix = '-'; + } + else { + number_set_number(mainwin_minus_num, 10); + t = time; + stime_prefix = ' '; + } + } + else { + number_set_number(mainwin_minus_num, 10); + t = time; + stime_prefix = ' '; + } + t /= 1000; + + /* Show the time in the format HH:MM when we have more than 100 + * minutes. */ + if (t >= 100 * 60) + t /= 60; + number_set_number(mainwin_10min_num, t / 600); + number_set_number(mainwin_min_num, (t / 60) % 10); + number_set_number(mainwin_10sec_num, (t / 10) % 6); + number_set_number(mainwin_sec_num, t % 10); + + if (!mainwin_sposition->hs_pressed) { + gchar *time_str; + + time_str = g_strdup_printf("%c%2.2d", stime_prefix, t / 60); + textbox_set_text(mainwin_stime_min, time_str); + g_free(time_str); + + time_str = g_strdup_printf("%2.2d", t % 60); + textbox_set_text(mainwin_stime_sec, time_str); + g_free(time_str); + } + + time /= 1000; + length /= 1000; + if (length > 0) { + if (time > length) { + hslider_set_position(mainwin_position, 219); + hslider_set_position(mainwin_sposition, 13); + } + /* update the slider position ONLY if there is not a seek in progress */ + else if (seek_state == MAINWIN_SEEK_NIL) { + hslider_set_position(mainwin_position, (time * 219) / length); + hslider_set_position(mainwin_sposition, + ((time * 12) / length) + 1); + } + } + else { + hslider_set_position(mainwin_position, 0); + hslider_set_position(mainwin_sposition, 1); + } +} + +static gboolean +mainwin_idle_func(gpointer data) +{ + static gint count = 0; + gint time = 0; + + /* run audcore events, then run our own. --nenolod */ + switch((time = audcore_generic_events())) + { + case -2: + /* no usable output device */ + GDK_THREADS_ENTER(); + run_no_output_device_dialog(); + mainwin_stop_pushed(); + GDK_THREADS_LEAVE(); + ev_waiting = FALSE; + break; + + default: + idle_func_update_song_info(time); + /* nothing at this time */ + } + + GDK_THREADS_ENTER(); + + if (playback_get_playing()) + vis_playback_start(); + else { + vis_playback_stop(); + ab_position_a = ab_position_b = -1; + } + + draw_main_window(mainwin_force_redraw); + + if (!count) { + read_volume(VOLSET_UPDATE); + count = 10; + } + else + count--; + + mainwin_force_redraw = FALSE; + draw_equalizer_window(FALSE); + draw_playlist_window(FALSE); + + if (mainwin_title_text) { + G_LOCK(mainwin_title); + gtk_window_set_title(GTK_WINDOW(mainwin), mainwin_title_text); + g_free(mainwin_title_text); + mainwin_title_text = NULL; + G_UNLOCK(mainwin_title); + + mainwin_set_info_text(); + playlistwin_update_list(playlist_get_active()); + } + + /* tristate buttons seek */ + if ( seek_state != MAINWIN_SEEK_NIL ) + { + GTimeVal now_time; + GTimeVal delta_time; + gulong now_dur; + g_get_current_time(&now_time); + + delta_time.tv_usec = now_time.tv_usec - cb_time.tv_usec; + delta_time.tv_sec = now_time.tv_sec - cb_time.tv_sec; + + now_dur = labs((delta_time.tv_sec * 1000) + (glong) (delta_time.tv_usec / 1000)); + + if ( now_dur > TRISTATE_THRESHOLD ) + { + gint np; + if (seek_state == MAINWIN_SEEK_REV) + np = seek_initial_pos - labs((gulong)(now_dur/100)); /* seek back */ + else + np = seek_initial_pos + labs((gulong)(now_dur/100)); /* seek forward */ + + /* boundaries check */ + if (np < 0 ) + np = 0; + else if ( np > 219 ) + np = 219; + + hslider_set_position( mainwin_position , np ); + mainwin_position_motion_cb( np ); + } + } + + GDK_THREADS_LEAVE(); + + /* + if (seek_state == MAINWIN_SEEK_REV) + playback_seek(CLAMP(playback_get_time() - 1000, 0, + playlist_get_current_length()) / 1000); + else if (seek_state == MAINWIN_SEEK_FWD) + playback_seek(CLAMP(playback_get_time() + 1000, 0, + playlist_get_current_length()) / 1000); + */ + + return TRUE; +} + + +/* toggleactionentries actions */ + +void +action_anamode_peaks( GtkToggleAction * action ) +{ + cfg.analyzer_peaks = gtk_toggle_action_get_active( action ); +} + +void +action_autoscroll_songname( GtkToggleAction * action ) +{ + mainwin_set_title_scroll(gtk_toggle_action_get_active(action)); + playlistwin_set_sinfo_scroll(cfg.autoscroll); /* propagate scroll setting to playlistwin_sinfo */ +} + +void +action_playback_noplaylistadvance( GtkToggleAction * action ) +{ + cfg.no_playlist_advance = gtk_toggle_action_get_active( action ); +} + +void +action_playback_repeat( GtkToggleAction * action ) +{ + cfg.repeat = gtk_toggle_action_get_active( action ); + tbutton_set_toggled(mainwin_repeat, cfg.repeat); +} + +void +action_playback_shuffle( GtkToggleAction * action ) +{ + cfg.shuffle = gtk_toggle_action_get_active( action ); + playlist_set_shuffle(cfg.shuffle); + tbutton_set_toggled(mainwin_shuffle, cfg.shuffle); +} + +void +action_stop_after_current_song( GtkToggleAction * action ) +{ + cfg.stopaftersong = gtk_toggle_action_get_active( action ); +} + +void +action_view_always_on_top( GtkToggleAction * action ) +{ + mainwin_menurow->mr_always_selected = gtk_toggle_action_get_active( action ); + cfg.always_on_top = mainwin_menurow->mr_always_selected; + widget_draw(WIDGET(mainwin_menurow)); + + if (starting_up == FALSE) + hint_set_always(cfg.always_on_top); +} + +void +action_view_doublesize( GtkToggleAction * action ) +{ + mainwin_menurow->mr_doublesize_selected = gtk_toggle_action_get_active( action ); + widget_draw(WIDGET(mainwin_menurow)); + set_doublesize(mainwin_menurow->mr_doublesize_selected); + gdk_flush(); +} + +void +action_view_easymove( GtkToggleAction * action ) +{ + cfg.easy_move = gtk_toggle_action_get_active( action ); +} + +void +action_view_on_all_workspaces( GtkToggleAction * action ) +{ + cfg.sticky = gtk_toggle_action_get_active( action ); + hint_set_sticky(cfg.sticky); +} + +void +action_roll_up_equalizer( GtkToggleAction * action ) +{ + equalizerwin_set_shade_menu_cb(gtk_toggle_action_get_active(action)); +} + +void +action_roll_up_player( GtkToggleAction * action ) +{ + mainwin_set_shade_menu_cb(gtk_toggle_action_get_active(action)); +} + +void +action_roll_up_playlist_editor( GtkToggleAction * action ) +{ + playlistwin_set_shade(gtk_toggle_action_get_active(action)); +} + +void +action_show_equalizer( GtkToggleAction * action ) +{ + if (gtk_toggle_action_get_active(action)) + equalizerwin_real_show(); + else + equalizerwin_real_hide(); +} + +void +action_show_playlist_editor( GtkToggleAction * action ) +{ + if (gtk_toggle_action_get_active(action)) + playlistwin_show(); + else + playlistwin_hide(); +} + +void +action_show_player( GtkToggleAction * action ) +{ + mainwin_show(gtk_toggle_action_get_active(action)); +} + + +/* radioactionentries actions (one callback for each radio group) */ + +void +action_anafoff( GtkAction *action, GtkRadioAction *current ) +{ + mainwin_vis_set_afalloff(gtk_radio_action_get_current_value(current)); +} + +void +action_anamode( GtkAction *action, GtkRadioAction *current ) +{ + mainwin_vis_set_analyzer_mode(gtk_radio_action_get_current_value(current)); +} + +void +action_anatype( GtkAction *action, GtkRadioAction *current ) +{ + mainwin_vis_set_analyzer_type(gtk_radio_action_get_current_value(current)); +} + +void +action_peafoff( GtkAction *action, GtkRadioAction *current ) +{ + mainwin_vis_set_pfalloff(gtk_radio_action_get_current_value(current)); +} + +void +action_refrate( GtkAction *action, GtkRadioAction *current ) +{ + mainwin_vis_set_refresh(gtk_radio_action_get_current_value(current)); +} + +void +action_scomode( GtkAction *action, GtkRadioAction *current ) +{ + cfg.scope_mode = gtk_radio_action_get_current_value(current); +} + +void +action_vismode( GtkAction *action, GtkRadioAction *current ) +{ + mainwin_vis_set_type_menu_cb(gtk_radio_action_get_current_value(current)); +} + +void +action_vprmode( GtkAction *action, GtkRadioAction *current ) +{ + cfg.voiceprint_mode = gtk_radio_action_get_current_value(current); +} + +void +action_wshmode( GtkAction *action, GtkRadioAction *current ) +{ + cfg.vu_mode = gtk_radio_action_get_current_value(current); +} + +void +action_viewtime( GtkAction *action, GtkRadioAction *current ) +{ + set_timer_mode_menu_cb(gtk_radio_action_get_current_value(current)); +} + + +/* actionentries actions */ + +void +action_about_audacious( void ) +{ + show_about_window(); +} + +void +action_play_file( void ) +{ + util_run_filebrowser(PLAY_BUTTON); +} + +void +action_play_location( void ) +{ + mainwin_show_add_url_window(); +} + +void +action_ab_set( void ) +{ + Playlist *playlist = playlist_get_active(); + if (playlist_get_current_length(playlist) != -1) + { + if (ab_position_a == -1) + { + ab_position_a = playback_get_time(); + ab_position_b = -1; + mainwin_lock_info_text("LOOP-POINT A POSITION SET."); + } + else if (ab_position_b == -1) + { + int time = playback_get_time(); + if (time > ab_position_a) + ab_position_b = time; + mainwin_release_info_text(); + } + else + { + ab_position_a = playback_get_time(); + ab_position_b = -1; + mainwin_lock_info_text("LOOP-POINT A POSITION RESET."); + } + } +} + +void +action_ab_clear( void ) +{ + Playlist *playlist = playlist_get_active(); + if (playlist_get_current_length(playlist) != -1) + { + ab_position_a = ab_position_b = -1; + mainwin_release_info_text(); + } +} + +void +action_current_track_info( void ) +{ + playlist_fileinfo_current(playlist_get_active()); +} + +void +action_jump_to_file( void ) +{ + mainwin_jump_to_file(); +} + +void +action_jump_to_playlist_start( void ) +{ + Playlist *playlist = playlist_get_active(); + playlist_set_position(playlist, 0); +} + +void +action_jump_to_time( void ) +{ + mainwin_jump_to_time(); +} + +void +action_playback_next( void ) +{ + Playlist *playlist = playlist_get_active(); + playlist_next(playlist); +} + +void +action_playback_previous( void ) +{ + Playlist *playlist = playlist_get_active(); + playlist_prev(playlist); +} + +void +action_playback_play( void ) +{ + mainwin_play_pushed(); +} + +void +action_playback_playcd( void ) +{ + play_medium(); +} + +void +action_playback_pause( void ) +{ + playback_pause(); +} + +void +action_playback_stop( void ) +{ + mainwin_stop_pushed(); +} + +void +action_preferences( void ) +{ + show_prefs_window(); +} + +void +action_quit( void ) +{ + mainwin_quit_cb(); +}