Mercurial > audlegacy
view audacious/ui_main.c @ 2272:9d8430979580 trunk
[svn] update copyright notice and make another string translateable
author | kiyoshi |
---|---|
date | Thu, 04 Jan 2007 15:30:25 -0800 |
parents | 8c9e27a834fd |
children | 74457ab2e9f1 |
line wrap: on
line source
/* 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 , "view roll up player" ); gtk_toggle_action_set_active( GTK_TOGGLE_ACTION(action) , shaded ); /* GtkWidget *widget = gtk_ui_manager_get_widget(ui_manager, "/mainwin-menus/main-menu/view/view roll up player" ); gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(widget), 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(NO_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(); }