/* X-Chat * Copyright (C) 2004-2007 Peter Zelezny. * * 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; either version 2 of the License, or * (at your option) any later version. * * 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 St, Fifth Floor, Boston, MA 02110-1301, USA */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/stat.h> #include <sys/types.h> #include "../common/hexchat.h" #include "../common/cfgfiles.h" #include "../common/fe.h" #include "../common/text.h" #include "../common/userlist.h" #include "../common/util.h" #include "../common/hexchatc.h" #include "../common/outbound.h" #include "fe-gtk.h" #include "gtkutil.h" #include "maingui.h" #include "palette.h" #include "pixmaps.h" #include "menu.h" #include "plugin-tray.h" #include "notifications/notification-backend.h" #ifdef WIN32 #include "../common/fe.h" #endif #include "sexy-spell-entry.h" GtkStyle *create_input_style (GtkStyle *); #define LABEL_INDENT 12 static int last_selected_page = 0; static int last_selected_row = 0; /* sound row */ static gboolean color_change; static struct hexchatprefs setup_prefs; static GtkWidget *cancel_button; static GtkWidget *font_dialog = NULL; enum { ST_END, ST_TOGGLE, ST_TOGGLR, ST_3OGGLE, ST_ENTRY, ST_EFONT, ST_EFILE, ST_EFOLDER, ST_MENU, ST_RADIO, ST_NUMBER, ST_HSCALE, ST_HEADER, ST_LABEL, ST_ALERTHEAD }; typedef struct { int type; char *label; int offset; char *tooltip; char const *const *list; int extra; } setting; #ifdef WIN32 static const char *const langsmenu[] = { N_("Afrikaans"), N_("Albanian"), N_("Amharic"), N_("Asturian"), N_("Azerbaijani"), N_("Basque"), N_("Belarusian"), N_("Bulgarian"), N_("Catalan"), N_("Chinese (Simplified)"), N_("Chinese (Traditional)"), N_("Czech"), N_("Danish"), N_("Dutch"), N_("English (British)"), N_("English"), N_("Estonian"), N_("Finnish"), N_("French"), N_("Galician"), N_("German"), N_("Greek"), N_("Gujarati"), N_("Hindi"), N_("Hungarian"), N_("Indonesian"), N_("Italian"), N_("Japanese"), N_("Kannada"), N_("Kinyarwanda"), N_("Korean"), N_("Latvian"), N_("Lithuanian"), N_("Macedonian"), N_("Malay"), N_("Malayalam"), N_("Norwegian (Bokmal)"), N_("Norwegian (Nynorsk)"), N_("Polish"), N_("Portuguese"), N_("Portuguese (Brazilian)"), N_("Punjabi"), N_("Russian"), N_("Serbian"), N_("Slovak"), N_("Slovenian"), N_("Spanish"), N_("Swedish"), N_("Thai"), N_("Turkish"), N_("Ukrainian"), N_("Vietnamese"), N_("Walloon"), NULL }; #endif static const setting appearance_settings[] = { {ST_HEADER, N_("General"),0,0,0}, #ifdef WIN32 {ST_MENU, N_("Language:"), P_OFFINTNL(hex_gui_lang), 0, langsmenu, 0}, {ST_EFONT, N_("Main font:"), P_OFFSETNL(hex_text_font_main), 0, 0, sizeof prefs.hex_text_font_main}, #else {ST_EFONT, N_("Font:"), P_OFFSETNL(hex_text_font), 0, 0, sizeof prefs.hex_text_font}, #endif {ST_HEADER, N_("Text Box"),0,0,0}, {ST_TOGGLE, N_("Colored nick names"), P_OFFINTNL(hex_text_color_nicks), N_("Give each person on IRC a different color"),0,0}, {ST_TOGGLR, N_("Indent nick names"), P_OFFINTNL(hex_text_indent), N_("Make nick names right-justified"),0,0}, {ST_TOGGLE, N_ ("Show marker line"), P_OFFINTNL (hex_text_show_marker), N_ ("Insert a red line after the last read text."), 0, 0}, {ST_EFILE, N_ ("Background image:"), P_OFFSETNL (hex_text_background), 0, 0, sizeof prefs.hex_text_background}, {ST_HEADER, N_("Transparency Settings"), 0,0,0}, {ST_HSCALE, N_("Window Opacity:"), P_OFFINTNL(hex_gui_transparency),0,0,0}, {ST_HEADER, N_("Time Stamps"),0,0,0}, {ST_TOGGLE, N_("Enable time stamps"), P_OFFINTNL(hex_stamp_text),0,0,1}, {ST_ENTRY, N_("Time stamp format:"), P_OFFSETNL(hex_stamp_text_format), #ifdef WIN32 N_("See the strftime MSDN article for details."),0,sizeof prefs.hex_stamp_text_format}, #else N_("See the strftime manpage for details."),0,sizeof prefs.hex_stamp_text_format}, #endif {ST_HEADER, N_("Title Bar"),0,0,0}, {ST_TOGGLE, N_("Show channel modes"), P_OFFINTNL(hex_gui_win_modes),0,0,0}, {ST_TOGGLR, N_("Show number of users"), P_OFFINTNL(hex_gui_win_ucount),0,0,0}, {ST_END, 0, 0, 0, 0, 0} }; static const char *const tabcompmenu[] = { N_("A-Z"), N_("Last-spoke order"), NULL }; static const setting inputbox_settings[] = { {ST_HEADER, N_("Input Box"),0,0,0}, {ST_TOGGLE, N_("Use the Text box font and colors"), P_OFFINTNL(hex_gui_input_style),0,0,0}, {ST_TOGGLE, N_("Render colors and attributes"), P_OFFINTNL (hex_gui_input_attr),0,0,0}, {ST_TOGGLE, N_("Show nick box"), P_OFFINTNL(hex_gui_input_nick),0,0,1}, {ST_TOGGLE, N_("Show user mode icon in nick box"), P_OFFINTNL(hex_gui_input_icon),0,0,0}, {ST_TOGGLE, N_("Spell checking"), P_OFFINTNL(hex_gui_input_spell),0,0,1}, {ST_ENTRY, N_("Dictionaries to use:"), P_OFFSETNL(hex_text_spell_langs),0,0,sizeof prefs.hex_text_spell_langs}, #ifdef WIN32 {ST_LABEL, N_("Use language codes (as in \"%LOCALAPPDATA%\\enchant\\myspell\\dicts\").\nSeparate multiple entries with commas.")}, #else {ST_LABEL, N_("Use language codes. Separate multiple entries with commas.")}, #endif {ST_HEADER, N_("Nick Completion"),0,0,0}, {ST_ENTRY, N_("Nick completion suffix:"), P_OFFSETNL(hex_completion_suffix),0,0,sizeof prefs.hex_completion_suffix}, {ST_MENU, N_("Nick completion sorted:"), P_OFFINTNL(hex_completion_sort), 0, tabcompmenu, 0}, {ST_NUMBER, N_("Nick completion amount:"), P_OFFINTNL(hex_completion_amount), N_("Threshold of nicks to start listing instead of completing"), (const char **)N_("nicks."), 1000}, {ST_END, 0, 0, 0, 0, 0} }; static const char *const lagmenutext[] = { N_("Off"), N_("Graphical"), N_("Text"), N_("Both"), NULL }; static const char *const ulmenutext[] = { N_("A-Z, Ops first"), N_("A-Z"), N_("Z-A, Ops last"), N_("Z-A"), N_("Unsorted"), NULL }; static const char *const cspos[] = { N_("Left (Upper)"), N_("Left (Lower)"), N_("Right (Upper)"), N_("Right (Lower)"), N_("Top"), N_("Bottom"), N_("Hidden"), NULL }; static const char *const ulpos[] = { N_("Left (Upper)"), N_("Left (Lower)"), N_("Right (Upper)"), N_("Right (Lower)"), NULL }; static const setting userlist_settings[] = { {ST_HEADER, N_("User List"),0,0,0}, {ST_TOGGLE, N_("Show hostnames in user list"), P_OFFINTNL(hex_gui_ulist_show_hosts), 0, 0, 0}, {ST_TOGGLE, N_("Use the Text box font and colors"), P_OFFINTNL(hex_gui_ulist_style),0,0,0}, {ST_TOGGLE, N_("Show icons for user modes"), P_OFFINTNL(hex_gui_ulist_icons), N_("Use graphical icons instead of text symbols in the user list."), 0, 0}, {ST_TOGGLE, N_("Color nicknames in userlist"), P_OFFINTNL(hex_gui_ulist_color), N_("Will color nicknames the same as in chat."), 0, 0}, {ST_TOGGLE, N_("Show user count in channels"), P_OFFINTNL(hex_gui_ulist_count), 0, 0, 0}, /* {ST_TOGGLE, N_("Resizable user list"), P_OFFINTNL(hex_gui_ulist_resizable),0,0,0},*/ {ST_MENU, N_("User list sorted by:"), P_OFFINTNL(hex_gui_ulist_sort), 0, ulmenutext, 0}, {ST_MENU, N_("Show user list at:"), P_OFFINTNL(hex_gui_ulist_pos), 0, ulpos, 1}, {ST_HEADER, N_("Away Tracking"),0,0,0}, {ST_TOGGLE, N_("Track the Away status of users and mark them in a different color"), P_OFFINTNL(hex_away_track),0,0,1}, {ST_NUMBER, N_("On channels smaller than:"), P_OFFINTNL(hex_away_size_max),0,0,10000}, {ST_HEADER, N_("Action Upon Double Click"),0,0,0}, {ST_ENTRY, N_("Execute command:"), P_OFFSETNL(hex_gui_ulist_doubleclick), 0, 0, sizeof prefs.hex_gui_ulist_doubleclick}, {ST_HEADER, N_("Extra Gadgets"),0,0,0}, {ST_MENU, N_("Lag meter:"), P_OFFINTNL(hex_gui_lagometer), 0, lagmenutext, 0}, {ST_MENU, N_("Throttle meter:"), P_OFFINTNL(hex_gui_throttlemeter), 0, lagmenutext, 0}, {ST_END, 0, 0, 0, 0, 0} }; static const char *const tabwin[] = { N_("Windows"), N_("Tabs"), NULL }; static const char *const focusnewtabsmenu[] = { N_("Never"), N_("Always"), N_("Only requested tabs"), NULL }; static const char *const noticeposmenu[] = { N_("Automatic"), N_("In an extra tab"), N_("In the front tab"), NULL }; static const char *const swtype[] = { N_("Tabs"), /* 0 tabs */ "", /* 1 reserved */ N_("Tree"), /* 2 tree */ NULL }; static const setting tabs_settings[] = { /*{ST_HEADER, N_("Channel Switcher"),0,0,0},*/ {ST_RADIO, N_("Switcher type:"),P_OFFINTNL(hex_gui_tab_layout), 0, swtype, 0}, {ST_TOGGLE, N_("Open an extra tab for server messages"), P_OFFINTNL(hex_gui_tab_server), 0, 0, 0}, {ST_TOGGLE, N_("Open a new tab when you receive a private message"), P_OFFINTNL(hex_gui_autoopen_dialog), 0, 0, 0}, {ST_TOGGLE, N_("Sort tabs in alphabetical order"), P_OFFINTNL(hex_gui_tab_sort), 0, 0, 0}, {ST_TOGGLE, N_("Show icons in the channel tree"), P_OFFINTNL(hex_gui_tab_icons), 0, 0, 0}, {ST_TOGGLE, N_("Show dotted lines in the channel tree"), P_OFFINTNL(hex_gui_tab_dots), 0, 0, 0}, {ST_TOGGLE, N_("Scroll mouse-wheel to change tabs"), P_OFFINTNL (hex_gui_tab_scrollchans), 0, 0, 0}, {ST_TOGGLE, N_("Middle click to close tab"), P_OFFINTNL(hex_gui_tab_middleclose), 0, 0, 0}, {ST_TOGGLE, N_("Smaller text"), P_OFFINTNL(hex_gui_tab_small), 0, 0, 0}, {ST_MENU, N_("Focus new tabs:"), P_OFFINTNL(hex_gui_tab_newtofront), 0, focusnewtabsmenu, 0}, {ST_MENU, N_("Placement of notices:"), P_OFFINTNL(hex_irc_notice_pos), 0, noticeposmenu, 0}, {ST_MENU, N_("Show channel switcher at:"), P_OFFINTNL(hex_gui_tab_pos), 0, cspos, 1}, {ST_NUMBER, N_("Shorten tab labels to:"), P_OFFINTNL(hex_gui_tab_trunc), 0, (const char **)N_("letters."), 99}, {ST_HEADER, N_("Tabs or Windows"),0,0,0}, {ST_MENU, N_("Open channels in:"), P_OFFINTNL(hex_gui_tab_chans), 0, tabwin, 0}, {ST_MENU, N_("Open dialogs in:"), P_OFFINTNL(hex_gui_tab_dialogs), 0, tabwin, 0}, {ST_MENU, N_("Open utilities in:"), P_OFFINTNL(hex_gui_tab_utils), N_("Open DCC, Ignore, Notify etc, in tabs or windows?"), tabwin, 0}, {ST_END, 0, 0, 0, 0, 0} }; static const setting color_settings[] = { {ST_TOGGLE, N_("Messages"), P_OFFINTNL(hex_text_stripcolor_msg), 0, 0, 0}, {ST_TOGGLE, N_("Scrollback"), P_OFFINTNL(hex_text_stripcolor_replay), 0, 0, 0}, {ST_TOGGLE, N_("Topic"), P_OFFINTNL(hex_text_stripcolor_topic), 0, 0, 0}, {ST_END, 0, 0, 0, 0, 0} }; static const char *const dccaccept[] = { N_("Ask for confirmation"), N_("Ask for download folder"), N_("Save without interaction"), NULL }; static const setting filexfer_settings[] = { {ST_HEADER, N_("Files and Directories"), 0, 0, 0}, {ST_MENU, N_("Auto accept file offers:"), P_OFFINTNL(hex_dcc_auto_recv), 0, dccaccept, 0}, {ST_EFOLDER,N_("Download files to:"), P_OFFSETNL(hex_dcc_dir), 0, 0, sizeof prefs.hex_dcc_dir}, {ST_EFOLDER,N_("Move completed files to:"), P_OFFSETNL(hex_dcc_completed_dir), 0, 0, sizeof prefs.hex_dcc_completed_dir}, {ST_TOGGLE, N_("Save nick name in filenames"), P_OFFINTNL(hex_dcc_save_nick), 0, 0, 0}, {ST_HEADER, N_("Auto Open DCC Windows"),0,0,0}, {ST_TOGGLE, N_("Send window"), P_OFFINTNL(hex_gui_autoopen_send), 0, 0, 0}, {ST_TOGGLE, N_("Receive window"), P_OFFINTNL(hex_gui_autoopen_recv), 0, 0, 0}, {ST_TOGGLE, N_("Chat window"), P_OFFINTNL(hex_gui_autoopen_chat), 0, 0, 0}, {ST_HEADER, N_("Maximum File Transfer Speeds (bytes per second)"), 0, 0, 0}, {ST_NUMBER, N_("One upload:"), P_OFFINTNL(hex_dcc_max_send_cps), N_("Maximum speed for one transfer"), 0, 10000000}, {ST_NUMBER, N_("One download:"), P_OFFINTNL(hex_dcc_max_get_cps), N_("Maximum speed for one transfer"), 0, 10000000}, {ST_NUMBER, N_("All uploads combined:"), P_OFFINTNL(hex_dcc_global_max_send_cps), N_("Maximum speed for all files"), 0, 10000000}, {ST_NUMBER, N_("All downloads combined:"), P_OFFINTNL(hex_dcc_global_max_get_cps), N_("Maximum speed for all files"), 0, 10000000}, {ST_END, 0, 0, 0, 0, 0} }; static const int balloonlist[3] = { P_OFFINTNL(hex_input_balloon_chans), P_OFFINTNL(hex_input_balloon_priv), P_OFFINTNL(hex_input_balloon_hilight) }; static const int trayblinklist[3] = { P_OFFINTNL(hex_input_tray_chans), P_OFFINTNL(hex_input_tray_priv), P_OFFINTNL(hex_input_tray_hilight) }; static const int taskbarlist[3] = { P_OFFINTNL(hex_input_flash_chans), P_OFFINTNL(hex_input_flash_priv), P_OFFINTNL(hex_input_flash_hilight) }; static const int beeplist[3] = { P_OFFINTNL(hex_input_beep_chans), P_OFFINTNL(hex_input_beep_priv), P_OFFINTNL(hex_input_beep_hilight) }; static const setting alert_settings[] = { {ST_HEADER, N_("Alerts"),0,0,0}, {ST_ALERTHEAD}, {ST_3OGGLE, N_("Show notifications on:"), 0, 0, (void *)balloonlist, 0}, {ST_3OGGLE, N_("Blink tray icon on:"), 0, 0, (void *)trayblinklist, 0}, {ST_3OGGLE, N_("Blink task bar on:"), 0, 0, (void *)taskbarlist, 0}, #ifdef WIN32 {ST_3OGGLE, N_("Make a beep sound on:"), 0, N_("Play the \"Instant Message Notification\" system sound upon the selected events"), (void *)beeplist, 0}, #else #ifdef USE_LIBCANBERRA {ST_3OGGLE, N_("Make a beep sound on:"), 0, N_("Play \"message-new-instant\" from the freedesktop.org sound theme upon the selected events"), (void *)beeplist, 0}, #else {ST_3OGGLE, N_("Make a beep sound on:"), 0, N_("Play a GTK beep upon the selected events"), (void *)beeplist, 0}, #endif #endif {ST_TOGGLE, N_("Omit alerts when marked as being away"), P_OFFINTNL(hex_away_omit_alerts), 0, 0, 0}, {ST_TOGGLE, N_("Omit alerts while the window is focused"), P_OFFINTNL(hex_gui_focus_omitalerts), 0, 0, 0}, {ST_HEADER, N_("Tray Behavior"), 0, 0, 0}, {ST_TOGGLE, N_("Enable system tray icon"), P_OFFINTNL(hex_gui_tray), 0, 0, 4}, {ST_TOGGLE, N_("Minimize to tray"), P_OFFINTNL(hex_gui_tray_minimize), 0, 0, 0}, {ST_TOGGLE, N_("Close to tray"), P_OFFINTNL(hex_gui_tray_close), 0, 0, 0}, {ST_TOGGLE, N_("Automatically mark away/back"), P_OFFINTNL(hex_gui_tray_away), N_("Automatically change status when hiding to tray."), 0, 0}, {ST_TOGGLE, N_("Only show notifications when hidden or iconified"), P_OFFINTNL(hex_gui_tray_quiet), 0, 0, 0}, {ST_HEADER, N_("Highlighted Messages"),0,0,0}, {ST_LABEL, N_("Highlighted messages are ones where your nickname is mentioned, but also:"), 0, 0, 0, 1}, {ST_ENTRY, N_("Extra words to highlight:"), P_OFFSETNL(hex_irc_extra_hilight), 0, 0, sizeof prefs.hex_irc_extra_hilight}, {ST_ENTRY, N_("Nick names not to highlight:"), P_OFFSETNL(hex_irc_no_hilight), 0, 0, sizeof prefs.hex_irc_no_hilight}, {ST_ENTRY, N_("Nick names to always highlight:"), P_OFFSETNL(hex_irc_nick_hilight), 0, 0, sizeof prefs.hex_irc_nick_hilight}, {ST_LABEL, N_("Separate multiple words with commas.\nWildcards are accepted.")}, {ST_END, 0, 0, 0, 0, 0} }; static const setting alert_settings_nonotifications[] = { {ST_HEADER, N_("Alerts"),0,0,0}, {ST_ALERTHEAD}, {ST_3OGGLE, N_("Blink tray icon on:"), 0, 0, (void *)trayblinklist, 0}, #ifdef HAVE_GTK_MAC {ST_3OGGLE, N_("Bounce dock icon on:"), 0, 0, (void *)taskbarlist, 0}, #else #ifndef __APPLE__ {ST_3OGGLE, N_("Blink task bar on:"), 0, 0, (void *)taskbarlist, 0}, #endif #endif #ifdef WIN32 {ST_3OGGLE, N_("Make a beep sound on:"), 0, N_("Play the \"Instant Message Notification\" system sound upon the selected events"), (void *)beeplist, 0}, #else #ifdef USE_LIBCANBERRA {ST_3OGGLE, N_("Make a beep sound on:"), 0, N_("Play \"message-new-instant\" from the freedesktop.org sound theme upon the selected events"), (void *)beeplist, 0}, #else {ST_3OGGLE, N_("Make a beep sound on:"), 0, N_("Play a GTK beep upon the selected events"), (void *)beeplist, 0}, #endif #endif {ST_TOGGLE, N_("Omit alerts when marked as being away"), P_OFFINTNL(hex_away_omit_alerts), 0, 0, 0}, {ST_TOGGLE, N_("Omit alerts while the window is focused"), P_OFFINTNL(hex_gui_focus_omitalerts), 0, 0, 0}, {ST_HEADER, N_("Tray Behavior"), 0, 0, 0}, {ST_TOGGLE, N_("Enable system tray icon"), P_OFFINTNL(hex_gui_tray), 0, 0, 4}, {ST_TOGGLE, N_("Minimize to tray"), P_OFFINTNL(hex_gui_tray_minimize), 0, 0, 0}, {ST_TOGGLE, N_("Close to tray"), P_OFFINTNL(hex_gui_tray_close), 0, 0, 0}, {ST_TOGGLE, N_("Automatically mark away/back"), P_OFFINTNL(hex_gui_tray_away), N_("Automatically change status when hiding to tray."), 0, 0}, {ST_HEADER, N_("Highlighted Messages"),0,0,0}, {ST_LABEL, N_("Highlighted messages are ones where your nickname is mentioned, but also:"), 0, 0, 0, 1}, {ST_ENTRY, N_("Extra words to highlight:"), P_OFFSETNL(hex_irc_extra_hilight), 0, 0, sizeof prefs.hex_irc_extra_hilight}, {ST_ENTRY, N_("Nick names not to highlight:"), P_OFFSETNL(hex_irc_no_hilight), 0, 0, sizeof prefs.hex_irc_no_hilight}, {ST_ENTRY, N_("Nick names to always highlight:"), P_OFFSETNL(hex_irc_nick_hilight), 0, 0, sizeof prefs.hex_irc_nick_hilight}, {ST_LABEL, N_("Separate multiple words with commas.\nWildcards are accepted.")}, {ST_END, 0, 0, 0, 0, 0} }; static const setting alert_settings_unity[] = { {ST_HEADER, N_("Alerts"),0,0,0}, {ST_ALERTHEAD}, {ST_3OGGLE, N_("Show notifications on:"), 0, 0, (void *)balloonlist, 0}, {ST_3OGGLE, N_("Blink task bar on:"), 0, 0, (void *)taskbarlist, 0}, {ST_3OGGLE, N_("Make a beep sound on:"), 0, 0, (void *)beeplist, 0}, {ST_TOGGLE, N_("Omit alerts when marked as being away"), P_OFFINTNL(hex_away_omit_alerts), 0, 0, 0}, {ST_TOGGLE, N_("Omit alerts while the window is focused"), P_OFFINTNL(hex_gui_focus_omitalerts), 0, 0, 0}, {ST_HEADER, N_("Highlighted Messages"),0,0,0}, {ST_LABEL, N_("Highlighted messages are ones where your nickname is mentioned, but also:"), 0, 0, 0, 1}, {ST_ENTRY, N_("Extra words to highlight:"), P_OFFSETNL(hex_irc_extra_hilight), 0, 0, sizeof prefs.hex_irc_extra_hilight}, {ST_ENTRY, N_("Nick names not to highlight:"), P_OFFSETNL(hex_irc_no_hilight), 0, 0, sizeof prefs.hex_irc_no_hilight}, {ST_ENTRY, N_("Nick names to always highlight:"), P_OFFSETNL(hex_irc_nick_hilight), 0, 0, sizeof prefs.hex_irc_nick_hilight}, {ST_LABEL, N_("Separate multiple words with commas.\nWildcards are accepted.")}, {ST_END, 0, 0, 0, 0, 0} }; static const setting alert_settings_unityandnonotifications[] = { {ST_HEADER, N_("Alerts"), 0, 0, 0}, {ST_ALERTHEAD}, {ST_3OGGLE, N_("Blink task bar on:"), 0, 0, (void *)taskbarlist, 0}, {ST_3OGGLE, N_("Make a beep sound on:"), 0, 0, (void *)beeplist, 0}, {ST_TOGGLE, N_("Omit alerts when marked as being away"), P_OFFINTNL (hex_away_omit_alerts), 0, 0, 0}, {ST_TOGGLE, N_("Omit alerts while the window is focused"), P_OFFINTNL (hex_gui_focus_omitalerts), 0, 0, 0}, {ST_HEADER, N_("Highlighted Messages"), 0, 0, 0}, {ST_LABEL, N_("Highlighted messages are ones where your nickname is mentioned, but also:"), 0, 0, 0, 1}, {ST_ENTRY, N_("Extra words to highlight:"), P_OFFSETNL (hex_irc_extra_hilight), 0, 0, sizeof prefs.hex_irc_extra_hilight}, {ST_ENTRY, N_("Nick names not to highlight:"), P_OFFSETNL (hex_irc_no_hilight), 0, 0, sizeof prefs.hex_irc_no_hilight}, {ST_ENTRY, N_("Nick names to always highlight:"), P_OFFSETNL (hex_irc_nick_hilight), 0, 0, sizeof prefs.hex_irc_nick_hilight}, {ST_LABEL, N_("Separate multiple words with commas.\nWildcards are accepted.")}, {ST_END, 0, 0, 0, 0, 0} }; static const setting general_settings[] = { {ST_HEADER, N_("Default Messages"),0,0,0}, {ST_ENTRY, N_("Quit:"), P_OFFSETNL(hex_irc_quit_reason), 0, 0, sizeof prefs.hex_irc_quit_reason}, {ST_ENTRY, N_("Leave channel:"), P_OFFSETNL(hex_irc_part_reason), 0, 0, sizeof prefs.hex_irc_part_reason}, {ST_ENTRY, N_("Away:"), P_OFFSETNL(hex_away_reason), 0, 0, sizeof prefs.hex_away_reason}, {ST_HEADER, N_("Away"),0,0,0}, {ST_TOGGLE, N_("Show away once"), P_OFFINTNL(hex_away_show_once), N_("Show identical away messages only once."), 0, 0}, {ST_TOGGLE, N_("Automatically unmark away"), P_OFFINTNL(hex_away_auto_unmark), N_("Unmark yourself as away before sending messages."), 0, 0}, {ST_HEADER, N_("Miscellaneous"),0,0,0}, {ST_TOGGLE, N_("Display MODEs in raw form"), P_OFFINTNL(hex_irc_raw_modes), 0, 0, 0}, {ST_TOGGLE, N_("WHOIS on notify"), P_OFFINTNL(hex_notify_whois_online), N_("Sends a /WHOIS when a user comes online in your notify list."), 0, 0}, {ST_TOGGLE, N_("Hide join and part messages"), P_OFFINTNL(hex_irc_conf_mode), N_("Hide channel join/part messages by default."), 0, 0}, {ST_TOGGLE, N_("Hide nick change messages"), P_OFFINTNL(hex_irc_hide_nickchange), 0, 0, 0}, {ST_END, 0, 0, 0, 0, 0} }; static const char *const bantypemenu[] = { N_("*!*@*.host"), N_("*!*@domain"), N_("*!*user@*.host"), N_("*!*user@domain"), NULL }; static const setting advanced_settings[] = { {ST_HEADER, N_("Auto Copy Behavior"),0,0,0}, {ST_TOGGLE, N_("Automatically copy selected text"), P_OFFINTNL(hex_text_autocopy_text), N_("Copy selected text to clipboard when left mouse button is released. " "Otherwise, CONTROL-SHIFT-C will copy the " "selected text to the clipboard."), 0, 0}, {ST_TOGGLE, N_("Automatically include time stamps"), P_OFFINTNL(hex_text_autocopy_stamp), N_("Automatically include time stamps in copied lines of text. Otherwise, " "include time stamps if the SHIFT key is held down while selecting."), 0, 0}, {ST_TOGGLE, N_("Automatically include color information"), P_OFFINTNL(hex_text_autocopy_color), N_("Automatically include color information in copied lines of text. " "Otherwise, include color information if the CONTROL key is held down " "while selecting."), 0, 0}, {ST_HEADER, N_("Miscellaneous"), 0, 0, 0}, {ST_ENTRY, N_("Real name:"), P_OFFSETNL(hex_irc_real_name), 0, 0, sizeof prefs.hex_irc_real_name}, #ifdef WIN32 {ST_ENTRY, N_("Alternative fonts:"), P_OFFSETNL(hex_text_font_alternative), N_("Separate multiple entries with commas without spaces before or after."), 0, sizeof prefs.hex_text_font_alternative}, #endif {ST_TOGGLE, N_("Display lists in compact mode"), P_OFFINTNL(hex_gui_compact), N_("Use less spacing between user list/channel tree rows."), 0, 0}, {ST_TOGGLE, N_("Use server time if supported"), P_OFFINTNL(hex_irc_cap_server_time), N_("Display timestamps obtained from server if it supports the time-server extension."), 0, 0}, {ST_TOGGLE, N_("Automatically reconnect to servers on disconnect"), P_OFFINTNL(hex_net_auto_reconnect), 0, 0, 1}, {ST_NUMBER, N_("Auto reconnect delay:"), P_OFFINTNL(hex_net_reconnect_delay), 0, 0, 9999}, {ST_NUMBER, N_("Auto join delay:"), P_OFFINTNL(hex_irc_join_delay), 0, 0, 9999}, {ST_MENU, N_("Ban Type:"), P_OFFINTNL(hex_irc_ban_type), N_("Attempt to use this banmask when banning or quieting. (requires irc_who_join)"), bantypemenu, 0}, {ST_END, 0, 0, 0, 0, 0} }; static const setting logging_settings[] = { {ST_HEADER, N_("Logging"),0,0,0}, {ST_TOGGLE, N_("Display scrollback from previous session"), P_OFFINTNL(hex_text_replay), 0, 0, 0}, {ST_NUMBER, N_("Scrollback lines:"), P_OFFINTNL(hex_text_max_lines),0,0,100000}, {ST_TOGGLE, N_("Enable logging of conversations to disk"), P_OFFINTNL(hex_irc_logging), 0, 0, 0}, {ST_ENTRY, N_("Log filename:"), P_OFFSETNL(hex_irc_logmask), 0, 0, sizeof prefs.hex_irc_logmask}, {ST_LABEL, N_("%s=Server %c=Channel %n=Network.")}, {ST_HEADER, N_("Time Stamps"),0,0,0}, {ST_TOGGLE, N_("Insert timestamps in logs"), P_OFFINTNL(hex_stamp_log), 0, 0, 1}, {ST_ENTRY, N_("Log timestamp format:"), P_OFFSETNL(hex_stamp_log_format), 0, 0, sizeof prefs.hex_stamp_log_format}, #ifdef WIN32 {ST_LABEL, N_("See the strftime MSDN article for details.")}, #else {ST_LABEL, N_("See the strftime manpage for details.")}, #endif {ST_HEADER, N_("URLs"),0,0,0}, {ST_TOGGLE, N_("Enable logging of URLs to disk"), P_OFFINTNL(hex_url_logging), 0, 0, 0}, {ST_TOGGLE, N_("Enable URL grabber"), P_OFFINTNL(hex_url_grabber), 0, 0, 1}, {ST_NUMBER, N_("Maximum number of URLs to grab:"), P_OFFINTNL(hex_url_grabber_limit), 0, 0, 9999}, {ST_END, 0, 0, 0, 0, 0} }; static const char *const proxytypes[] = { N_("(Disabled)"), N_("Wingate"), N_("Socks4"), N_("Socks5"), N_("HTTP"), #ifdef USE_LIBPROXY N_("Auto"), #endif NULL }; static const char *const proxyuse[] = { N_("All Connections"), N_("IRC Server Only"), N_("DCC Get Only"), NULL }; static const setting network_settings[] = { {ST_HEADER, N_("Your Address"), 0, 0, 0, 0}, {ST_ENTRY, N_("Bind to:"), P_OFFSETNL(hex_net_bind_host), 0, 0, sizeof prefs.hex_net_bind_host}, {ST_LABEL, N_("Only useful for computers with multiple addresses.")}, {ST_HEADER, N_("File Transfers"), 0, 0, 0}, {ST_TOGGLE, N_("Get my address from the IRC server"), P_OFFINTNL(hex_dcc_ip_from_server), N_("Asks the IRC server for your real address. Use this if you have a 192.168.*.* address!"), 0, 0}, {ST_ENTRY, N_("DCC IP address:"), P_OFFSETNL(hex_dcc_ip), N_("Claim you are at this address when offering files."), 0, sizeof prefs.hex_dcc_ip}, {ST_NUMBER, N_("First DCC send port:"), P_OFFINTNL(hex_dcc_port_first), 0, 0, 65535}, {ST_NUMBER, N_("Last DCC send port:"), P_OFFINTNL(hex_dcc_port_last), 0, (const char **)N_("!Leave ports at zero for full range."), 65535}, {ST_HEADER, N_("Proxy Server"), 0, 0, 0, 0}, {ST_ENTRY, N_("Hostname:"), P_OFFSETNL(hex_net_proxy_host), 0, 0, sizeof prefs.hex_net_proxy_host}, {ST_NUMBER, N_("Port:"), P_OFFINTNL(hex_net_proxy_port), 0, 0, 65535}, {ST_MENU, N_("Type:"), P_OFFINTNL(hex_net_proxy_type), 0, proxytypes, 0}, {ST_MENU, N_("Use proxy for:"), P_OFFINTNL(hex_net_proxy_use), 0, proxyuse, 0}, {ST_HEADER, N_("Proxy Authentication"), 0, 0, 0, 0}, {ST_TOGGLE, N_("Use Authentication (HTTP or Socks5 only)"), P_OFFINTNL(hex_net_proxy_auth), 0, 0, 0}, {ST_ENTRY, N_("Username:"), P_OFFSETNL(hex_net_proxy_user), 0, 0, sizeof prefs.hex_net_proxy_user}, {ST_ENTRY, N_("Password:"), P_OFFSETNL(hex_net_proxy_pass), 0, GINT_TO_POINTER(1), sizeof prefs.hex_net_proxy_pass}, {ST_END, 0, 0, 0, 0, 0} }; static const setting identd_settings[] = { {ST_HEADER, N_("Identd Server"), 0, 0, 0, 0}, {ST_TOGGLE, N_("Enabled"), P_OFFINTNL(hex_identd_server), N_("Server will respond with the networks username"), 0, 1}, {ST_NUMBER, N_("Port:"), P_OFFINTNL(hex_identd_port), N_("You must have permissions to listen on this port. " "If not 113 (0 defaults to this) then you must configure port-forwarding."), 0, 65535}, {ST_END, 0, 0, 0, 0, 0} }; #define setup_get_str(pr,set) (((char *)pr)+set->offset) #define setup_get_int(pr,set) *(((int *)pr)+set->offset) #define setup_get_int3(pr,off) *(((int *)pr)+off) #define setup_set_int(pr,set,num) *((int *)pr+set->offset)=num #define setup_set_str(pr,set,str) strcpy(((char *)pr)+set->offset,str) static void setup_3oggle_cb (GtkToggleButton *but, unsigned int *setting) { *setting = gtk_toggle_button_get_active (but); } static void setup_headlabel (GtkWidget *tab, int row, int col, char *text) { GtkWidget *label; char buf[128]; char *sp; g_snprintf (buf, sizeof (buf), "<b><span size=\"smaller\">%s</span></b>", text); sp = strchr (buf + 17, ' '); if (sp) *sp = '\n'; label = gtk_label_new (NULL); gtk_label_set_markup (GTK_LABEL (label), buf); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); gtk_table_attach (GTK_TABLE (tab), label, col, col + 1, row, row + 1, 0, 0, 4, 0); } static void setup_create_alert_header (GtkWidget *tab, int row, const setting *set) { setup_headlabel (tab, row, 3, _("Channel Message")); setup_headlabel (tab, row, 4, _("Private Message")); setup_headlabel (tab, row, 5, _("Highlighted Message")); } /* makes 3 toggles side-by-side */ static void setup_create_3oggle (GtkWidget *tab, int row, const setting *set) { GtkWidget *label, *wid; int *offsets = (int *)set->list; label = gtk_label_new (_(set->label)); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); if (set->tooltip) { gtk_widget_set_tooltip_text (label, _(set->tooltip)); } gtk_table_attach (GTK_TABLE (tab), label, 2, 3, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, LABEL_INDENT, 0); wid = gtk_check_button_new (); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (wid), setup_get_int3 (&setup_prefs, offsets[0])); g_signal_connect (G_OBJECT (wid), "toggled", G_CALLBACK (setup_3oggle_cb), ((int *)&setup_prefs) + offsets[0]); gtk_table_attach (GTK_TABLE (tab), wid, 3, 4, row, row + 1, 0, 0, 0, 0); wid = gtk_check_button_new (); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (wid), setup_get_int3 (&setup_prefs, offsets[1])); g_signal_connect (G_OBJECT (wid), "toggled", G_CALLBACK (setup_3oggle_cb), ((int *)&setup_prefs) + offsets[1]); gtk_table_attach (GTK_TABLE (tab), wid, 4, 5, row, row + 1, 0, 0, 0, 0); wid = gtk_check_button_new (); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (wid), setup_get_int3 (&setup_prefs, offsets[2])); g_signal_connect (G_OBJECT (wid), "toggled", G_CALLBACK (setup_3oggle_cb), ((int *)&setup_prefs) + offsets[2]); gtk_table_attach (GTK_TABLE (tab), wid, 5, 6, row, row + 1, 0, 0, 0, 0); } static void setup_toggle_cb (GtkToggleButton *but, const setting *set) { GtkWidget *label, *disable_wid; setup_set_int (&setup_prefs, set, gtk_toggle_button_get_active (but)); /* does this toggle also enable/disable another widget? */ disable_wid = g_object_get_data (G_OBJECT (but), "nxt"); if (disable_wid) { gtk_widget_set_sensitive (disable_wid, gtk_toggle_button_get_active (but)); label = g_object_get_data (G_OBJECT (disable_wid), "lbl"); gtk_widget_set_sensitive (label, gtk_toggle_button_get_active (but)); } } static void setup_toggle_sensitive_cb (GtkToggleButton *but, GtkWidget *wid) { gtk_widget_set_sensitive (wid, gtk_toggle_button_get_active (but)); } static void setup_create_toggleR (GtkWidget *tab, int row, const setting *set) { GtkWidget *wid; wid = gtk_check_button_new_with_label (_(set->label)); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (wid), setup_get_int (&setup_prefs, set)); g_signal_connect (G_OBJECT (wid), "toggled", G_CALLBACK (setup_toggle_cb), (gpointer)set); if (set->tooltip) gtk_widget_set_tooltip_text (wid, _(set->tooltip)); gtk_table_attach (GTK_TABLE (tab), wid, 4, 5, row, row + 1, GTK_EXPAND | GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0); } static GtkWidget * setup_create_toggleL (GtkWidget *tab, int row, const setting *set) { GtkWidget *wid; wid = gtk_check_button_new_with_label (_(set->label)); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (wid), setup_get_int (&setup_prefs, set)); g_signal_connect (G_OBJECT (wid), "toggled", G_CALLBACK (setup_toggle_cb), (gpointer)set); if (set->tooltip) gtk_widget_set_tooltip_text (wid, _(set->tooltip)); gtk_table_attach (GTK_TABLE (tab), wid, 2, row==6 ? 6 : 4, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, LABEL_INDENT, 0); return wid; } static GtkWidget * setup_create_italic_label (char *text) { GtkWidget *label; char buf[256]; label = gtk_label_new (NULL); g_snprintf (buf, sizeof (buf), "<i><span size=\"smaller\">%s</span></i>", text); gtk_label_set_markup (GTK_LABEL (label), buf); gtk_label_set_justify (GTK_LABEL (label), GTK_JUSTIFY_CENTER); return label; } static void setup_spin_cb (GtkSpinButton *spin, const setting *set) { setup_set_int (&setup_prefs, set, gtk_spin_button_get_value_as_int (spin)); } static GtkWidget * setup_create_spin (GtkWidget *table, int row, const setting *set) { GtkWidget *label, *wid, *rbox, *align; char *text; label = gtk_label_new (_(set->label)); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); gtk_table_attach (GTK_TABLE (table), label, 2, 3, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, LABEL_INDENT, 0); align = gtk_alignment_new (0.0, 0.5, 0.0, 0.0); gtk_table_attach (GTK_TABLE (table), align, 3, 4, row, row + 1, GTK_EXPAND | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0); rbox = gtk_hbox_new (0, 0); gtk_container_add (GTK_CONTAINER (align), rbox); wid = gtk_spin_button_new_with_range (0, set->extra, 1); g_object_set_data (G_OBJECT (wid), "lbl", label); if (set->tooltip) gtk_widget_set_tooltip_text (wid, _(set->tooltip)); gtk_spin_button_set_value (GTK_SPIN_BUTTON (wid), setup_get_int (&setup_prefs, set)); g_signal_connect (G_OBJECT (wid), "value_changed", G_CALLBACK (setup_spin_cb), (gpointer)set); gtk_box_pack_start (GTK_BOX (rbox), wid, 0, 0, 0); if (set->list) { text = _((char *)set->list); if (text[0] == '!') label = setup_create_italic_label (text + 1); else label = gtk_label_new (text); gtk_box_pack_start (GTK_BOX (rbox), label, 0, 0, 6); } return wid; } static gint setup_apply_trans (int *tag) { prefs.hex_gui_transparency = setup_prefs.hex_gui_transparency; gtk_window_set_opacity (GTK_WINDOW (current_sess->gui->window), (prefs.hex_gui_transparency / 255.)); /* mg_update_xtext (current_sess->gui->xtext); */ *tag = 0; return 0; } static void setup_hscale_cb (GtkHScale *wid, const setting *set) { static int tag = 0; setup_set_int (&setup_prefs, set, (int) gtk_range_get_value (GTK_RANGE (wid))); if (tag == 0) { tag = g_idle_add ((GSourceFunc) setup_apply_trans, &tag); } } static void setup_create_hscale (GtkWidget *table, int row, const setting *set) { GtkWidget *wid; wid = gtk_label_new (_(set->label)); gtk_misc_set_alignment (GTK_MISC (wid), 0.0, 0.5); gtk_table_attach (GTK_TABLE (table), wid, 2, 3, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, LABEL_INDENT, 0); wid = gtk_hscale_new_with_range (0., 255., 1.); gtk_scale_set_value_pos (GTK_SCALE (wid), GTK_POS_RIGHT); gtk_range_set_value (GTK_RANGE (wid), setup_get_int (&setup_prefs, set)); g_signal_connect (G_OBJECT(wid), "value_changed", G_CALLBACK (setup_hscale_cb), (gpointer)set); gtk_table_attach (GTK_TABLE (table), wid, 3, 6, row, row + 1, GTK_EXPAND | GTK_FILL, GTK_FILL, 0, 0); #ifndef WIN32 /* Windows always supports this */ /* Only used for transparency currently */ if (!gtk_widget_is_composited (current_sess->gui->window)) gtk_widget_set_sensitive (wid, FALSE); #endif } static GtkWidget *proxy_user; /* username GtkEntry */ static GtkWidget *proxy_pass; /* password GtkEntry */ static void setup_menu_cb (GtkWidget *cbox, const setting *set) { int n = gtk_combo_box_get_active (GTK_COMBO_BOX (cbox)); /* set the prefs.<field> */ setup_set_int (&setup_prefs, set, n + set->extra); if (set->list == proxytypes) { /* only HTTP and Socks5 can use a username/pass */ gtk_widget_set_sensitive (proxy_user, (n == 3 || n == 4 || n == 5)); gtk_widget_set_sensitive (proxy_pass, (n == 3 || n == 4 || n == 5)); } } static void setup_radio_cb (GtkWidget *item, const setting *set) { if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (item))) { int n = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (item), "n")); /* set the prefs.<field> */ setup_set_int (&setup_prefs, set, n); } } static int setup_create_radio (GtkWidget *table, int row, const setting *set) { GtkWidget *wid, *hbox; int i; const char **text = (const char **)set->list; GSList *group; wid = gtk_label_new (_(set->label)); gtk_misc_set_alignment (GTK_MISC (wid), 0.0, 0.5); gtk_table_attach (GTK_TABLE (table), wid, 2, 3, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, LABEL_INDENT, 0); hbox = gtk_hbox_new (0, 0); gtk_table_attach (GTK_TABLE (table), hbox, 3, 4, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0); i = 0; group = NULL; while (text[i]) { if (text[i][0] != 0) { wid = gtk_radio_button_new_with_mnemonic (group, _(text[i])); /*if (set->tooltip) gtk_widget_set_tooltip_text (wid, _(set->tooltip));*/ group = gtk_radio_button_get_group (GTK_RADIO_BUTTON (wid)); gtk_container_add (GTK_CONTAINER (hbox), wid); if (i == setup_get_int (&setup_prefs, set)) gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (wid), TRUE); g_object_set_data (G_OBJECT (wid), "n", GINT_TO_POINTER (i)); g_signal_connect (G_OBJECT (wid), "toggled", G_CALLBACK (setup_radio_cb), (gpointer)set); } i++; row++; } return i; } /* static const char *id_strings[] = { "", "*", "%C4*%C18%B%B", "%U" }; static void setup_id_menu_cb (GtkWidget *item, char *dest) { int n = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (item), "n")); strcpy (dest, id_strings[n]); } static void setup_create_id_menu (GtkWidget *table, char *label, int row, char *dest) { GtkWidget *wid, *menu, *item; int i, def = 0; static const char *text[] = { ("(disabled)"), ("A star (*)"), ("A red star (*)"), ("Underlined") }; wid = gtk_label_new (label); gtk_misc_set_alignment (GTK_MISC (wid), 0.0, 0.5); gtk_table_attach (GTK_TABLE (table), wid, 2, 3, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, LABEL_INDENT, 0); wid = gtk_option_menu_new (); menu = gtk_menu_new (); for (i = 0; i < 4; i++) { if (strcmp (id_strings[i], dest) == 0) { def = i; break; } } i = 0; while (text[i]) { item = gtk_menu_item_new_with_label (_(text[i])); g_object_set_data (G_OBJECT (item), "n", GINT_TO_POINTER (i)); gtk_widget_show (item); gtk_menu_shell_append (GTK_MENU_SHELL (menu), item); g_signal_connect (G_OBJECT (item), "activate", G_CALLBACK (setup_id_menu_cb), dest); i++; } gtk_option_menu_set_menu (GTK_OPTION_MENU (wid), menu); gtk_option_menu_set_history (GTK_OPTION_MENU (wid), def); gtk_table_attach (GTK_TABLE (table), wid, 3, 4, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0); } */ static void setup_create_menu (GtkWidget *table, int row, const setting *set) { GtkWidget *wid, *cbox, *box; const char **text = (const char **)set->list; int i; wid = gtk_label_new (_(set->label)); gtk_misc_set_alignment (GTK_MISC (wid), 0.0, 0.5); gtk_table_attach (GTK_TABLE (table), wid, 2, 3, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, LABEL_INDENT, 0); cbox = gtk_combo_box_text_new (); for (i = 0; text[i]; i++) gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT (cbox), _(text[i])); gtk_combo_box_set_active (GTK_COMBO_BOX (cbox), setup_get_int (&setup_prefs, set) - set->extra); g_signal_connect (G_OBJECT (cbox), "changed", G_CALLBACK (setup_menu_cb), (gpointer)set); box = gtk_hbox_new (0, 0); gtk_box_pack_start (GTK_BOX (box), cbox, 0, 0, 0); gtk_table_attach (GTK_TABLE (table), box, 3, 4, row, row + 1, GTK_EXPAND | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0); } static void setup_filereq_cb (GtkWidget *entry, char *file) { if (file) { if (file[0]) gtk_entry_set_text (GTK_ENTRY (entry), file); } } static void setup_browsefile_cb (GtkWidget *button, GtkWidget *entry) { /* used for background image only */ char *filter; int filter_type; #ifdef WIN32 filter = "*png;*.tiff;*.gif;*.jpeg;*.jpg"; filter_type = FRF_EXTENSIONS; #else filter = "image/*"; filter_type = FRF_MIMETYPES; #endif gtkutil_file_req (_("Select an Image File"), setup_filereq_cb, entry, NULL, filter, filter_type|FRF_RECENTLYUSED); } static void setup_fontsel_destroy (GtkWidget *button, GtkFontSelectionDialog *dialog) { font_dialog = NULL; } static void setup_fontsel_cb (GtkWidget *button, GtkFontSelectionDialog *dialog) { GtkWidget *entry; char *font_name; entry = g_object_get_data (G_OBJECT (button), "e"); font_name = gtk_font_selection_dialog_get_font_name (dialog); gtk_entry_set_text (GTK_ENTRY (entry), font_name); g_free (font_name); gtk_widget_destroy (GTK_WIDGET (dialog)); font_dialog = NULL; } static void setup_fontsel_cancel (GtkWidget *button, GtkFontSelectionDialog *dialog) { gtk_widget_destroy (GTK_WIDGET (dialog)); font_dialog = NULL; } static void setup_browsefolder_cb (GtkWidget *button, GtkEntry *entry) { gtkutil_file_req (_("Select Download Folder"), setup_filereq_cb, entry, (char*)gtk_entry_get_text (entry), NULL, FRF_CHOOSEFOLDER); } static void setup_browsefont_cb (GtkWidget *button, GtkWidget *entry) { GtkFontSelection *sel; GtkFontSelectionDialog *dialog; GtkWidget *ok_button; dialog = (GtkFontSelectionDialog *) gtk_font_selection_dialog_new (_("Select font")); font_dialog = (GtkWidget *)dialog; /* global var */ sel = (GtkFontSelection *) gtk_font_selection_dialog_get_font_selection (dialog); if (gtk_entry_get_text (GTK_ENTRY (entry))[0]) gtk_font_selection_set_font_name (sel, gtk_entry_get_text (GTK_ENTRY (entry))); ok_button = gtk_font_selection_dialog_get_ok_button (dialog); g_object_set_data (G_OBJECT (ok_button), "e", entry); g_signal_connect (G_OBJECT (dialog), "destroy", G_CALLBACK (setup_fontsel_destroy), dialog); g_signal_connect (G_OBJECT (ok_button), "clicked", G_CALLBACK (setup_fontsel_cb), dialog); g_signal_connect (G_OBJECT (gtk_font_selection_dialog_get_cancel_button (dialog)), "clicked", G_CALLBACK (setup_fontsel_cancel), dialog); gtk_widget_show (GTK_WIDGET (dialog)); } static void setup_entry_cb (GtkEntry *entry, setting *set) { int size; int pos; unsigned char *p = (unsigned char*)gtk_entry_get_text (entry); int len = strlen (p); /* need to truncate? */ if (len >= set->extra) { len = pos = 0; while (1) { size = g_utf8_skip [*p]; len += size; p += size; /* truncate to "set->extra" BYTES */ if (len >= set->extra) { gtk_editable_delete_text (GTK_EDITABLE (entry), pos, -1); break; } pos++; } } else { setup_set_str (&setup_prefs, set, gtk_entry_get_text (entry)); } } static void setup_create_label (GtkWidget *table, int row, const setting *set) { gtk_table_attach (GTK_TABLE (table), setup_create_italic_label (_(set->label)), set->extra ? 1 : 3, 5, row, row + 1, GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0); } static GtkWidget * setup_create_entry (GtkWidget *table, int row, const setting *set) { GtkWidget *label; GtkWidget *wid, *bwid; label = gtk_label_new (_(set->label)); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); gtk_table_attach (GTK_TABLE (table), label, 2, 3, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, LABEL_INDENT, 0); wid = gtk_entry_new (); g_object_set_data (G_OBJECT (wid), "lbl", label); if (set->list) gtk_entry_set_visibility (GTK_ENTRY (wid), FALSE); if (set->tooltip) gtk_widget_set_tooltip_text (wid, _(set->tooltip)); gtk_entry_set_max_length (GTK_ENTRY (wid), set->extra - 1); gtk_entry_set_text (GTK_ENTRY (wid), setup_get_str (&setup_prefs, set)); g_signal_connect (G_OBJECT (wid), "changed", G_CALLBACK (setup_entry_cb), (gpointer)set); if (set->offset == P_OFFSETNL(hex_net_proxy_user)) proxy_user = wid; if (set->offset == P_OFFSETNL(hex_net_proxy_pass)) proxy_pass = wid; /* only http and Socks5 can auth */ if ( (set->offset == P_OFFSETNL(hex_net_proxy_pass) || set->offset == P_OFFSETNL(hex_net_proxy_user)) && (setup_prefs.hex_net_proxy_type != 4 && setup_prefs.hex_net_proxy_type != 3 && setup_prefs.hex_net_proxy_type != 5) ) gtk_widget_set_sensitive (wid, FALSE); if (set->type == ST_ENTRY) gtk_table_attach (GTK_TABLE (table), wid, 3, 6, row, row + 1, GTK_EXPAND | GTK_FILL, GTK_FILL, 0, 0); else { gtk_table_attach (GTK_TABLE (table), wid, 3, 5, row, row + 1, GTK_EXPAND | GTK_FILL, GTK_SHRINK, 0, 0); bwid = gtk_button_new_with_label (_("Browse...")); gtk_table_attach (GTK_TABLE (table), bwid, 5, 6, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_FILL, 0, 0); if (set->type == ST_EFILE) g_signal_connect (G_OBJECT (bwid), "clicked", G_CALLBACK (setup_browsefile_cb), wid); if (set->type == ST_EFONT) g_signal_connect (G_OBJECT (bwid), "clicked", G_CALLBACK (setup_browsefont_cb), wid); if (set->type == ST_EFOLDER) g_signal_connect (G_OBJECT (bwid), "clicked", G_CALLBACK (setup_browsefolder_cb), wid); } return wid; } static void setup_create_header (GtkWidget *table, int row, char *labeltext) { GtkWidget *label; char buf[128]; if (row == 0) g_snprintf (buf, sizeof (buf), "<b>%s</b>", _(labeltext)); else g_snprintf (buf, sizeof (buf), "\n<b>%s</b>", _(labeltext)); label = gtk_label_new (NULL); gtk_label_set_markup (GTK_LABEL (label), buf); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); gtk_table_attach (GTK_TABLE (table), label, 0, 4, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 5); } static void setup_create_button (GtkWidget *table, int row, char *label, GCallback callback) { GtkWidget *but = gtk_button_new_with_label (label); gtk_table_attach (GTK_TABLE (table), but, 2, 3, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 5); g_signal_connect (G_OBJECT (but), "clicked", callback, NULL); } static GtkWidget * setup_create_frame (void) { GtkWidget *tab; tab = gtk_table_new (3, 2, FALSE); gtk_container_set_border_width (GTK_CONTAINER (tab), 6); gtk_table_set_row_spacings (GTK_TABLE (tab), 2); gtk_table_set_col_spacings (GTK_TABLE (tab), 3); return tab; } static void open_data_cb (GtkWidget *button, gpointer data) { fe_open_url (get_xdir ()); } static GtkWidget * setup_create_page (const setting *set) { int i, row, do_disable; GtkWidget *tab; GtkWidget *wid = NULL, *parentwid = NULL; tab = setup_create_frame (); gtk_container_set_border_width (GTK_CONTAINER (tab), 6); i = row = do_disable = 0; while (set[i].type != ST_END) { switch (set[i].type) { case ST_HEADER: setup_create_header (tab, row, set[i].label); break; case ST_EFONT: case ST_ENTRY: case ST_EFILE: case ST_EFOLDER: wid = setup_create_entry (tab, row, &set[i]); break; case ST_TOGGLR: row--; setup_create_toggleR (tab, row, &set[i]); break; case ST_TOGGLE: wid = setup_create_toggleL (tab, row, &set[i]); if (set[i].extra) do_disable = set[i].extra; break; case ST_3OGGLE: setup_create_3oggle (tab, row, &set[i]); break; case ST_MENU: setup_create_menu (tab, row, &set[i]); break; case ST_RADIO: row += setup_create_radio (tab, row, &set[i]); break; case ST_NUMBER: wid = setup_create_spin (tab, row, &set[i]); break; case ST_HSCALE: setup_create_hscale (tab, row, &set[i]); break; case ST_LABEL: setup_create_label (tab, row, &set[i]); break; case ST_ALERTHEAD: setup_create_alert_header (tab, row, &set[i]); } if (do_disable) { if (GTK_IS_WIDGET (parentwid)) { g_signal_connect (G_OBJECT (parentwid), "toggled", G_CALLBACK(setup_toggle_sensitive_cb), wid); gtk_widget_set_sensitive (wid, gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (parentwid))); do_disable--; if (!do_disable) parentwid = NULL; } else parentwid = wid; } i++; row++; } if (set == logging_settings) { setup_create_button (tab, row, _("Open Data Folder"), G_CALLBACK(open_data_cb)); } return tab; } static void setup_color_ok_cb (GtkWidget *button, GtkWidget *dialog) { GtkColorSelectionDialog *cdialog = GTK_COLOR_SELECTION_DIALOG (dialog); GdkColor *col; GdkColor old_color; GtkStyle *style; col = g_object_get_data (G_OBJECT (button), "c"); old_color = *col; button = g_object_get_data (G_OBJECT (button), "b"); if (!GTK_IS_WIDGET (button)) { gtk_widget_destroy (dialog); return; } color_change = TRUE; gtk_color_selection_get_current_color (GTK_COLOR_SELECTION (gtk_color_selection_dialog_get_color_selection (cdialog)), col); gdk_colormap_alloc_color (gtk_widget_get_colormap (button), col, TRUE, TRUE); style = gtk_style_new (); style->bg[0] = *col; gtk_widget_set_style (button, style); g_object_unref (style); /* is this line correct?? */ gdk_colormap_free_colors (gtk_widget_get_colormap (button), &old_color, 1); gtk_widget_destroy (dialog); } static void setup_color_cb (GtkWidget *button, gpointer userdata) { GtkWidget *dialog, *cancel_button, *ok_button, *help_button; GtkColorSelectionDialog *cdialog; GdkColor *color; color = &colors[GPOINTER_TO_INT (userdata)]; dialog = gtk_color_selection_dialog_new (_("Select color")); cdialog = GTK_COLOR_SELECTION_DIALOG (dialog); g_object_get (G_OBJECT(cdialog), "cancel-button", &cancel_button, "ok-button", &ok_button, "help-button", &help_button, NULL); gtk_widget_hide (help_button); g_signal_connect (G_OBJECT (ok_button), "clicked", G_CALLBACK (setup_color_ok_cb), dialog); g_signal_connect (G_OBJECT (cancel_button), "clicked", G_CALLBACK (gtkutil_destroy), dialog); g_object_set_data (G_OBJECT (ok_button), "c", color); g_object_set_data (G_OBJECT (ok_button), "b", button); gtk_widget_set_sensitive (help_button, FALSE); gtk_color_selection_set_current_color (GTK_COLOR_SELECTION (gtk_color_selection_dialog_get_color_selection (cdialog)), color); gtk_widget_show (dialog); g_object_unref (cancel_button); g_object_unref (ok_button); g_object_unref (help_button); } static void setup_create_color_button (GtkWidget *table, int num, int row, int col) { GtkWidget *but; GtkStyle *style; char buf[64]; if (num > 31) strcpy (buf, "<span size=\"x-small\"> </span>"); else /* 12345678901 23456789 01 23456789 */ sprintf (buf, "<span size=\"x-small\">%d</span>", num); but = gtk_button_new_with_label (" "); gtk_label_set_markup (GTK_LABEL (gtk_bin_get_child (GTK_BIN (but))), buf); /* win32 build uses this to turn off themeing */ g_object_set_data (G_OBJECT (but), "hexchat-color", (gpointer)1); gtk_table_attach (GTK_TABLE (table), but, col, col+1, row, row+1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, 0, 0); g_signal_connect (G_OBJECT (but), "clicked", G_CALLBACK (setup_color_cb), GINT_TO_POINTER (num)); style = gtk_style_new (); style->bg[GTK_STATE_NORMAL] = colors[num]; gtk_widget_set_style (but, style); g_object_unref (style); } static void setup_create_other_colorR (char *text, int num, int row, GtkWidget *tab) { GtkWidget *label; label = gtk_label_new (text); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); gtk_table_attach (GTK_TABLE (tab), label, 5, 9, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, LABEL_INDENT, 0); setup_create_color_button (tab, num, row, 9); } static void setup_create_other_color (char *text, int num, int row, GtkWidget *tab) { GtkWidget *label; label = gtk_label_new (text); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); gtk_table_attach (GTK_TABLE (tab), label, 2, 3, row, row + 1, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, LABEL_INDENT, 0); setup_create_color_button (tab, num, row, 3); } static GtkWidget * setup_create_color_page (void) { GtkWidget *tab, *box, *label; int i; box = gtk_vbox_new (FALSE, 0); gtk_container_set_border_width (GTK_CONTAINER (box), 6); tab = gtk_table_new (9, 2, FALSE); gtk_container_set_border_width (GTK_CONTAINER (tab), 6); gtk_table_set_row_spacings (GTK_TABLE (tab), 2); gtk_table_set_col_spacings (GTK_TABLE (tab), 3); gtk_container_add (GTK_CONTAINER (box), tab); setup_create_header (tab, 0, N_("Text Colors")); label = gtk_label_new (_("mIRC colors:")); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); gtk_table_attach (GTK_TABLE (tab), label, 2, 3, 1, 2, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, LABEL_INDENT, 0); for (i = 0; i < 16; i++) setup_create_color_button (tab, i, 1, i+3); label = gtk_label_new (_("Local colors:")); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); gtk_table_attach (GTK_TABLE (tab), label, 2, 3, 2, 3, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, LABEL_INDENT, 0); for (i = 16; i < 32; i++) setup_create_color_button (tab, i, 2, (i+3) - 16); setup_create_other_color (_("Foreground:"), COL_FG, 3, tab); setup_create_other_colorR (_("Background:"), COL_BG, 3, tab); setup_create_header (tab, 5, N_("Selected Text")); setup_create_other_color (_("Foreground:"), COL_MARK_FG, 6, tab); setup_create_other_colorR (_("Background:"), COL_MARK_BG, 6, tab); setup_create_header (tab, 8, N_("Interface Colors")); setup_create_other_color (_("New data:"), COL_NEW_DATA, 9, tab); setup_create_other_colorR (_("Marker line:"), COL_MARKER, 9, tab); setup_create_other_color (_("New message:"), COL_NEW_MSG, 10, tab); setup_create_other_colorR (_("Away user:"), COL_AWAY, 10, tab); setup_create_other_color (_("Highlight:"), COL_HILIGHT, 11, tab); setup_create_other_colorR (_("Spell checker:"), COL_SPELL, 11, tab); setup_create_header (tab, 15, N_("Color Stripping")); /* label = gtk_label_new (_("Strip colors from:")); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); gtk_table_attach (GTK_TABLE (tab), label, 2, 3, 16, 17, GTK_SHRINK | GTK_FILL, GTK_SHRINK | GTK_FILL, LABEL_INDENT, 0); */ for (i = 0; i < 3; i++) { setup_create_toggleL (tab, i + 16, &color_settings[i]); } return box; } /* === GLOBALS for sound GUI === */ static GtkWidget *sndfile_entry; static int ignore_changed = FALSE; extern struct text_event te[]; /* text.c */ extern char *sound_files[]; static void setup_snd_populate (GtkTreeView * treeview) { GtkListStore *store; GtkTreeIter iter; GtkTreeSelection *sel; GtkTreePath *path; int i; sel = gtk_tree_view_get_selection (treeview); store = (GtkListStore *)gtk_tree_view_get_model (treeview); for (i = NUM_XP-1; i >= 0; i--) { gtk_list_store_prepend (store, &iter); if (sound_files[i]) gtk_list_store_set (store, &iter, 0, te[i].name, 1, sound_files[i], 2, i, -1); else gtk_list_store_set (store, &iter, 0, te[i].name, 1, "", 2, i, -1); if (i == last_selected_row) { gtk_tree_selection_select_iter (sel, &iter); path = gtk_tree_model_get_path (GTK_TREE_MODEL (store), &iter); if (path) { gtk_tree_view_scroll_to_cell (treeview, path, NULL, TRUE, 0.5, 0.5); gtk_tree_view_set_cursor (treeview, path, NULL, FALSE); gtk_tree_path_free (path); } } } } static int setup_snd_get_selected (GtkTreeSelection *sel, GtkTreeIter *iter) { int n; GtkTreeModel *model; if (!gtk_tree_selection_get_selected (sel, &model, iter)) return -1; gtk_tree_model_get (model, iter, 2, &n, -1); return n; } static void setup_snd_row_cb (GtkTreeSelection *sel, gpointer user_data) { int n; GtkTreeIter iter; n = setup_snd_get_selected (sel, &iter); if (n == -1) return; last_selected_row = n; ignore_changed = TRUE; if (sound_files[n]) gtk_entry_set_text (GTK_ENTRY (sndfile_entry), sound_files[n]); else gtk_entry_set_text (GTK_ENTRY (sndfile_entry), ""); ignore_changed = FALSE; } static void setup_snd_add_columns (GtkTreeView * treeview) { GtkCellRenderer *renderer; GtkTreeModel *model; /* event column */ renderer = gtk_cell_renderer_text_new (); gtk_tree_view_insert_column_with_attributes (GTK_TREE_VIEW (treeview), -1, _("Event"), renderer, "text", 0, NULL); /* file column */ renderer = gtk_cell_renderer_text_new (); gtk_tree_view_insert_column_with_attributes (GTK_TREE_VIEW (treeview), -1, _("Sound file"), renderer, "text", 1, NULL); model = GTK_TREE_MODEL (gtk_list_store_new (3, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_INT)); gtk_tree_view_set_model (GTK_TREE_VIEW (treeview), model); g_object_unref (model); } static void setup_snd_filereq_cb (GtkWidget *entry, char *file) { if (file) { if (file[0]) { /* Use just the filename if the given sound file is in the default <config>/sounds directory. * We're comparing absolute paths so this won't work in portable mode which uses a relative path. */ if (!strcmp (g_path_get_dirname (file), g_build_filename (get_xdir (), HEXCHAT_SOUND_DIR, NULL))) { gtk_entry_set_text (GTK_ENTRY (entry), g_path_get_basename (file)); } else { gtk_entry_set_text (GTK_ENTRY (entry), file); } } } } static void setup_snd_browse_cb (GtkWidget *button, GtkEntry *entry) { char *sounds_dir = g_build_filename (get_xdir (), HEXCHAT_SOUND_DIR, NULL); char *filter = NULL; int filter_type; #ifdef WIN32 /* win32 only supports wav, others could support anything */ filter = "*.wav"; filter_type = FRF_EXTENSIONS; #else filter = "audio/*"; filter_type = FRF_MIMETYPES; #endif gtkutil_file_req (_("Select a sound file"), setup_snd_filereq_cb, entry, sounds_dir, filter, FRF_FILTERISINITIAL|filter_type); g_free (sounds_dir); } static void setup_snd_play_cb (GtkWidget *button, GtkEntry *entry) { sound_play (gtk_entry_get_text (entry), FALSE); } static void setup_snd_changed_cb (GtkEntry *ent, GtkTreeView *tree) { int n; GtkTreeIter iter; GtkListStore *store; GtkTreeSelection *sel; if (ignore_changed) return; sel = gtk_tree_view_get_selection (tree); n = setup_snd_get_selected (sel, &iter); if (n == -1) return; /* get the new sound file */ g_free (sound_files[n]); sound_files[n] = g_strdup (gtk_entry_get_text (GTK_ENTRY (ent))); /* update the TreeView list */ store = (GtkListStore *)gtk_tree_view_get_model (tree); gtk_list_store_set (store, &iter, 1, sound_files[n], -1); gtk_widget_set_sensitive (cancel_button, FALSE); } static GtkWidget * setup_create_sound_page (void) { GtkWidget *vbox1; GtkWidget *vbox2; GtkWidget *scrolledwindow1; GtkWidget *sound_tree; GtkWidget *table1; GtkWidget *sound_label; GtkWidget *sound_browse; GtkWidget *sound_play; GtkTreeSelection *sel; vbox1 = gtk_vbox_new (FALSE, 0); gtk_container_set_border_width (GTK_CONTAINER (vbox1), 6); gtk_widget_show (vbox1); vbox2 = gtk_vbox_new (FALSE, 0); gtk_widget_show (vbox2); gtk_container_add (GTK_CONTAINER (vbox1), vbox2); scrolledwindow1 = gtk_scrolled_window_new (NULL, NULL); gtk_widget_show (scrolledwindow1); gtk_container_add (GTK_CONTAINER (vbox2), scrolledwindow1); gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolledwindow1), GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS); gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (scrolledwindow1), GTK_SHADOW_IN); sound_tree = gtk_tree_view_new (); sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (sound_tree)); gtk_tree_selection_set_mode (sel, GTK_SELECTION_SINGLE); setup_snd_add_columns (GTK_TREE_VIEW (sound_tree)); setup_snd_populate (GTK_TREE_VIEW (sound_tree)); g_signal_connect (G_OBJECT (sel), "changed", G_CALLBACK (setup_snd_row_cb), NULL); gtk_widget_show (sound_tree); gtk_container_add (GTK_CONTAINER (scrolledwindow1), sound_tree); gtk_tree_view_set_rules_hint (GTK_TREE_VIEW (sound_tree), TRUE); table1 = gtk_table_new (2, 3, FALSE); gtk_widget_show (table1); gtk_box_pack_start (GTK_BOX (vbox2), table1, FALSE, TRUE, 8); gtk_table_set_row_spacings (GTK_TABLE (table1), 2); gtk_table_set_col_spacings (GTK_TABLE (table1), 4); sound_label = gtk_label_new_with_mnemonic (_("Sound file:")); gtk_widget_show (sound_label); gtk_table_attach (GTK_TABLE (table1), sound_label, 0, 1, 0, 1, (GtkAttachOptions) (GTK_FILL), (GtkAttachOptions) (0), 0, 0); gtk_misc_set_alignment (GTK_MISC (sound_label), 0, 0.5); sndfile_entry = gtk_entry_new (); g_signal_connect (G_OBJECT (sndfile_entry), "changed", G_CALLBACK (setup_snd_changed_cb), sound_tree); gtk_widget_show (sndfile_entry); gtk_table_attach (GTK_TABLE (table1), sndfile_entry, 0, 1, 1, 2, (GtkAttachOptions) (GTK_EXPAND | GTK_FILL), (GtkAttachOptions) (0), 0, 0); sound_browse = gtk_button_new_with_mnemonic (_("_Browse...")); g_signal_connect (G_OBJECT (sound_browse), "clicked", G_CALLBACK (setup_snd_browse_cb), sndfile_entry); gtk_widget_show (sound_browse); gtk_table_attach (GTK_TABLE (table1), sound_browse, 1, 2, 1, 2, (GtkAttachOptions) (GTK_FILL), (GtkAttachOptions) (0), 0, 0); #ifdef GTK_STOCK_MEDIA_PLAY sound_play = gtk_button_new_from_stock (GTK_STOCK_MEDIA_PLAY); #else sound_play = gtk_button_new_with_mnemonic (_("_Play")); #endif g_signal_connect (G_OBJECT (sound_play), "clicked", G_CALLBACK (setup_snd_play_cb), sndfile_entry); gtk_widget_show (sound_play); gtk_table_attach (GTK_TABLE (table1), sound_play, 2, 3, 1, 2, (GtkAttachOptions) (GTK_FILL), (GtkAttachOptions) (0), 0, 0); setup_snd_row_cb (sel, NULL); return vbox1; } static void setup_add_page (const char *title, GtkWidget *book, GtkWidget *tab) { GtkWidget *label, *vvbox; char buf[128]; vvbox = gtk_vbox_new (FALSE, 0); /* label */ label = gtk_label_new (NULL); g_snprintf (buf, sizeof (buf), "<b><big>%s</big></b>", _(title)); gtk_label_set_markup (GTK_LABEL (label), buf); gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); gtk_misc_set_padding (GTK_MISC (label), 2, 1); gtk_box_pack_start (GTK_BOX (vvbox), label, FALSE, FALSE, 2); gtk_container_add (GTK_CONTAINER (vvbox), tab); gtk_notebook_append_page (GTK_NOTEBOOK (book), vvbox, NULL); } static const char *const cata[] = { N_("Interface"), N_("Appearance"), N_("Input box"), N_("User list"), N_("Channel switcher"), N_("Colors"), NULL, N_("Chatting"), N_("General"), N_("Alerts"), N_("Sounds"), N_("Logging"), N_("Advanced"), NULL, N_("Network"), N_("Network setup"), N_("File transfers"), N_("Identd"), NULL, NULL }; static GtkWidget * setup_create_pages (GtkWidget *box) { GtkWidget *book; book = gtk_notebook_new (); setup_add_page (cata[1], book, setup_create_page (appearance_settings)); setup_add_page (cata[2], book, setup_create_page (inputbox_settings)); setup_add_page (cata[3], book, setup_create_page (userlist_settings)); setup_add_page (cata[4], book, setup_create_page (tabs_settings)); setup_add_page (cata[5], book, setup_create_color_page ()); setup_add_page (cata[8], book, setup_create_page (general_settings)); if (unity_mode () && !notification_backend_supported ()) { setup_add_page (cata[9], book, setup_create_page (alert_settings_unityandnonotifications)); } else if (unity_mode ()) { setup_add_page (cata[9], book, setup_create_page (alert_settings_unity)); } else if (!notification_backend_supported ()) { setup_add_page (cata[9], book, setup_create_page (alert_settings_nonotifications)); } else { setup_add_page (cata[9], book, setup_create_page (alert_settings)); } setup_add_page (cata[10], book, setup_create_sound_page ()); setup_add_page (cata[11], book, setup_create_page (logging_settings)); setup_add_page (cata[12], book, setup_create_page (advanced_settings)); setup_add_page (cata[15], book, setup_create_page (network_settings)); setup_add_page (cata[16], book, setup_create_page (filexfer_settings)); setup_add_page (cata[17], book, setup_create_page (identd_settings)); gtk_notebook_set_show_tabs (GTK_NOTEBOOK (book), FALSE); gtk_notebook_set_show_border (GTK_NOTEBOOK (book), FALSE); gtk_container_add (GTK_CONTAINER (box), book); return book; } static void setup_tree_cb (GtkTreeView *treeview, GtkWidget *book) { GtkTreeSelection *selection = gtk_tree_view_get_selection (treeview); GtkTreeIter iter; GtkTreeModel *model; int page; if (gtk_tree_selection_get_selected (selection, &model, &iter)) { gtk_tree_model_get (model, &iter, 1, &page, -1); if (page != -1) { gtk_notebook_set_current_page (GTK_NOTEBOOK (book), page); last_selected_page = page; } } } static gboolean setup_tree_select_filter (GtkTreeSelection *selection, GtkTreeModel *model, GtkTreePath *path, gboolean path_selected, gpointer data) { if (gtk_tree_path_get_depth (path) > 1) return TRUE; return FALSE; } static void setup_create_tree (GtkWidget *box, GtkWidget *book) { GtkWidget *tree; GtkWidget *frame; GtkTreeStore *model; GtkTreeIter iter; GtkTreeIter child_iter; GtkTreeIter *sel_iter = NULL; GtkCellRenderer *renderer; GtkTreeSelection *sel; int i, page; model = gtk_tree_store_new (2, G_TYPE_STRING, G_TYPE_INT); i = 0; page = 0; do { gtk_tree_store_append (model, &iter, NULL); gtk_tree_store_set (model, &iter, 0, _(cata[i]), 1, -1, -1); i++; do { gtk_tree_store_append (model, &child_iter, &iter); gtk_tree_store_set (model, &child_iter, 0, _(cata[i]), 1, page, -1); if (page == last_selected_page) sel_iter = gtk_tree_iter_copy (&child_iter); page++; i++; } while (cata[i]); i++; } while (cata[i]); tree = gtk_tree_view_new_with_model (GTK_TREE_MODEL (model)); g_object_unref (G_OBJECT (model)); sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (tree)); gtk_tree_selection_set_mode (sel, GTK_SELECTION_BROWSE); gtk_tree_selection_set_select_function (sel, setup_tree_select_filter, NULL, NULL); g_signal_connect (G_OBJECT (tree), "cursor_changed", G_CALLBACK (setup_tree_cb), book); renderer = gtk_cell_renderer_text_new (); gtk_tree_view_insert_column_with_attributes (GTK_TREE_VIEW (tree), -1, _("Categories"), renderer, "text", 0, NULL); gtk_tree_view_expand_all (GTK_TREE_VIEW (tree)); frame = gtk_frame_new (NULL); gtk_container_add (GTK_CONTAINER (frame), tree); gtk_box_pack_start (GTK_BOX (box), frame, 0, 0, 0); gtk_box_reorder_child (GTK_BOX (box), frame, 0); if (sel_iter) { gtk_tree_selection_select_iter (sel, sel_iter); gtk_tree_iter_free (sel_iter); } } static void setup_apply_entry_style (GtkWidget *entry) { gtk_widget_modify_base (entry, GTK_STATE_NORMAL, &colors[COL_BG]); gtk_widget_modify_text (entry, GTK_STATE_NORMAL, &colors[COL_FG]); gtk_widget_modify_font (entry, input_style->font_desc); } static void setup_apply_to_sess (session_gui *gui) { mg_update_xtext (gui->xtext); if (prefs.hex_gui_ulist_style) gtk_widget_set_style (gui->user_tree, input_style); if (prefs.hex_gui_input_style) { extern char cursor_color_rc[]; char buf[256]; sprintf (buf, cursor_color_rc, (colors[COL_FG].red >> 8), (colors[COL_FG].green >> 8), (colors[COL_FG].blue >> 8)); gtk_rc_parse_string (buf); setup_apply_entry_style (gui->input_box); setup_apply_entry_style (gui->limit_entry); setup_apply_entry_style (gui->key_entry); setup_apply_entry_style (gui->topic_entry); } if (prefs.hex_gui_ulist_buttons) gtk_widget_show (gui->button_box); else gtk_widget_hide (gui->button_box); /* update active languages */ sexy_spell_entry_deactivate_language((SexySpellEntry *)gui->input_box,NULL); sexy_spell_entry_activate_default_languages((SexySpellEntry *)gui->input_box); sexy_spell_entry_set_checked ((SexySpellEntry *)gui->input_box, prefs.hex_gui_input_spell); sexy_spell_entry_set_parse_attributes ((SexySpellEntry *)gui->input_box, prefs.hex_gui_input_attr); } static void unslash (char *dir) { if (dir[0]) { int len = strlen (dir) - 1; #ifdef WIN32 if (dir[len] == '/' || dir[len] == '\\') #else if (dir[len] == '/') #endif dir[len] = 0; } } void setup_apply_real (int new_pix, int do_ulist, int do_layout, int do_identd) { GSList *list; session *sess; int done_main = FALSE; /* remove trailing slashes */ unslash (prefs.hex_dcc_dir); unslash (prefs.hex_dcc_completed_dir); g_mkdir (prefs.hex_dcc_dir, 0700); g_mkdir (prefs.hex_dcc_completed_dir, 0700); if (new_pix) { if (channelwin_pix) g_object_unref (channelwin_pix); channelwin_pix = pixmap_load_from_file (prefs.hex_text_background); } input_style = create_input_style (input_style); list = sess_list; while (list) { sess = list->data; if (sess->gui->is_tab) { /* only apply to main tabwindow once */ if (!done_main) { done_main = TRUE; setup_apply_to_sess (sess->gui); } } else { setup_apply_to_sess (sess->gui); } log_open_or_close (sess); if (do_ulist) userlist_rehash (sess); list = list->next; } mg_apply_setup (); tray_apply_setup (); if (do_layout) menu_change_layout (); if (do_identd) handle_command (current_sess, "IDENTD reload", FALSE); } static void setup_apply (struct hexchatprefs *pr) { #ifdef WIN32 PangoFontDescription *old_desc; PangoFontDescription *new_desc; char buffer[4 * FONTNAMELEN + 1]; #endif int new_pix = FALSE; int noapply = FALSE; int do_ulist = FALSE; int do_layout = FALSE; int do_identd = FALSE; if (strcmp (pr->hex_text_background, prefs.hex_text_background) != 0) new_pix = TRUE; #define DIFF(a) (pr->a != prefs.a) #ifdef WIN32 if (DIFF (hex_gui_lang)) noapply = TRUE; #endif if (DIFF (hex_gui_compact)) noapply = TRUE; if (DIFF (hex_gui_input_icon)) noapply = TRUE; if (DIFF (hex_gui_input_nick)) noapply = TRUE; if (DIFF (hex_gui_lagometer)) noapply = TRUE; if (DIFF (hex_gui_tab_icons)) noapply = TRUE; if (DIFF (hex_gui_tab_server)) noapply = TRUE; if (DIFF (hex_gui_tab_small)) noapply = TRUE; if (DIFF (hex_gui_tab_sort)) noapply = TRUE; if (DIFF (hex_gui_tab_trunc)) noapply = TRUE; if (DIFF (hex_gui_throttlemeter)) noapply = TRUE; if (DIFF (hex_gui_ulist_count)) noapply = TRUE; if (DIFF (hex_gui_ulist_icons)) noapply = TRUE; if (DIFF (hex_gui_ulist_resizable)) noapply = TRUE; if (DIFF (hex_gui_ulist_show_hosts)) noapply = TRUE; if (DIFF (hex_gui_ulist_style)) noapply = TRUE; if (DIFF (hex_gui_ulist_sort)) noapply = TRUE; if (DIFF (hex_gui_tab_dots)) do_layout = TRUE; if (DIFF (hex_gui_tab_layout)) do_layout = TRUE; if (DIFF (hex_identd_server) || DIFF (hex_identd_port)) do_identd = TRUE; if (color_change || (DIFF (hex_gui_ulist_color)) || (DIFF (hex_away_size_max)) || (DIFF (hex_away_track))) do_ulist = TRUE; if ((pr->hex_gui_tab_pos == 5 || pr->hex_gui_tab_pos == 6) && pr->hex_gui_tab_layout == 2 && pr->hex_gui_tab_pos != prefs.hex_gui_tab_pos) fe_message (_("You cannot place the tree on the top or bottom!\n" "Please change to the <b>Tabs</b> layout in the <b>View</b>" " menu first."), FE_MSG_WARN | FE_MSG_MARKUP); /* format cannot be blank, there is already a setting for this */ if (pr->hex_stamp_text_format[0] == 0) { pr->hex_stamp_text = 0; strcpy (pr->hex_stamp_text_format, prefs.hex_stamp_text_format); } memcpy (&prefs, pr, sizeof (prefs)); #ifdef WIN32 /* merge hex_font_main and hex_font_alternative into hex_font_normal */ old_desc = pango_font_description_from_string (prefs.hex_text_font_main); sprintf (buffer, "%s,%s", pango_font_description_get_family (old_desc), prefs.hex_text_font_alternative); new_desc = pango_font_description_from_string (buffer); pango_font_description_set_weight (new_desc, pango_font_description_get_weight (old_desc)); pango_font_description_set_style (new_desc, pango_font_description_get_style (old_desc)); pango_font_description_set_size (new_desc, pango_font_description_get_size (old_desc)); sprintf (prefs.hex_text_font, "%s", pango_font_description_to_string (new_desc)); /* FIXME this is not required after pango_font_description_from_string() g_free (old_desc); g_free (new_desc); */ #endif if (prefs.hex_irc_real_name[0] == 0) { fe_message (_("The Real name option cannot be left blank. Falling back to \"realname\"."), FE_MSG_WARN); strcpy (prefs.hex_irc_real_name, "realname"); } setup_apply_real (new_pix, do_ulist, do_layout, do_identd); if (noapply) fe_message (_("Some settings were changed that require a" " restart to take full effect."), FE_MSG_WARN); #ifndef WIN32 if (prefs.hex_dcc_auto_recv == 2) /* Auto */ { if (!strcmp ((char *)g_get_home_dir (), prefs.hex_dcc_dir)) { fe_message (_("*WARNING*\n" "Auto accepting DCC to your home directory\n" "can be dangerous and is exploitable. Eg:\n" "Someone could send you a .bash_profile"), FE_MSG_WARN); } } #endif } static void setup_ok_cb (GtkWidget *but, GtkWidget *win) { gtk_widget_destroy (win); setup_apply (&setup_prefs); save_config (); palette_save (); } static GtkWidget * setup_window_open (void) { GtkWidget *wid, *win, *vbox, *hbox, *hbbox; win = gtkutil_window_new (_(DISPLAY_NAME": Preferences"), "prefs", 0, 0, 2); vbox = gtk_vbox_new (FALSE, 5); gtk_container_set_border_width (GTK_CONTAINER (vbox), 6); gtk_container_add (GTK_CONTAINER (win), vbox); hbox = gtk_hbox_new (FALSE, 4); gtk_container_add (GTK_CONTAINER (vbox), hbox); setup_create_tree (hbox, setup_create_pages (hbox)); /* prepare the button box */ hbbox = gtk_hbutton_box_new (); gtk_button_box_set_layout (GTK_BUTTON_BOX (hbbox), GTK_BUTTONBOX_END); gtk_box_set_spacing (GTK_BOX (hbbox), 4); gtk_box_pack_end (GTK_BOX (vbox), hbbox, FALSE, FALSE, 0); cancel_button = wid = gtk_button_new_from_stock (GTK_STOCK_CANCEL); g_signal_connect (G_OBJECT (wid), "clicked", G_CALLBACK (gtkutil_destroy), win); gtk_box_pack_start (GTK_BOX (hbbox), wid, FALSE, FALSE, 0); wid = gtk_button_new_from_stock (GTK_STOCK_OK); g_signal_connect (G_OBJECT (wid), "clicked", G_CALLBACK (setup_ok_cb), win); gtk_box_pack_start (GTK_BOX (hbbox), wid, FALSE, FALSE, 0); gtk_widget_show_all (win); return win; } static void setup_close_cb (GtkWidget *win, GtkWidget **swin) { *swin = NULL; if (font_dialog) { gtk_widget_destroy (font_dialog); font_dialog = NULL; } } void setup_open (void) { static GtkWidget *setup_window = NULL; if (setup_window) { gtk_window_present (GTK_WINDOW (setup_window)); return; } memcpy (&setup_prefs, &prefs, sizeof (prefs)); color_change = FALSE; setup_window = setup_window_open (); g_signal_connect (G_OBJECT (setup_window), "destroy", G_CALLBACK (setup_close_cb), &setup_window); }