mirror of
https://github.com/moparisthebest/hexchat
synced 2024-11-10 11:25:08 -05:00
2261 lines
68 KiB
C
2261 lines
68 KiB
C
/* 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 "fe-gtk.h"
|
|
#include "gtkutil.h"
|
|
#include "maingui.h"
|
|
#include "palette.h"
|
|
#include "pixmaps.h"
|
|
#include "menu.h"
|
|
#include "plugin-tray.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_("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},
|
|
#if !defined (WIN32) && !defined (__APPLE__)
|
|
{ST_3OGGLE, N_("Show tray balloons on:"), 0, 0, (void *)balloonlist, 0},
|
|
#endif
|
|
{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},
|
|
#ifdef WIN32
|
|
{ST_TOGGLE, N_("Enable system tray icon"), P_OFFINTNL(hex_gui_tray), 0, 0, 3},
|
|
#else
|
|
{ST_TOGGLE, N_("Enable system tray icon"), P_OFFINTNL(hex_gui_tray), 0, 0, 4},
|
|
#endif
|
|
{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},
|
|
#ifndef WIN32
|
|
{ST_TOGGLE, N_("Only show tray balloons when hidden or iconified"), P_OFFINTNL(hex_gui_tray_quiet), 0, 0, 0},
|
|
#endif
|
|
|
|
{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 tray balloons 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 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_MSPROXY
|
|
N_("MS Proxy (ISA)"),
|
|
#endif
|
|
#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},
|
|
#ifdef USE_MSPROXY
|
|
{ST_TOGGLE, N_("Use Authentication (MS Proxy, HTTP or Socks5 only)"), P_OFFINTNL(hex_net_proxy_auth), 0, 0, 0},
|
|
#else
|
|
{ST_TOGGLE, N_("Use Authentication (HTTP or Socks5 only)"), P_OFFINTNL(hex_net_proxy_auth), 0, 0, 0},
|
|
#endif
|
|
{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}
|
|
};
|
|
|
|
#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;
|
|
|
|
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);
|
|
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;
|
|
int len = gtk_entry_get_text_length (entry);
|
|
unsigned char *p = (unsigned char*)gtk_entry_get_text (entry);
|
|
|
|
/* 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)
|
|
snprintf (buf, sizeof (buf), "<b>%s</b>", _(labeltext));
|
|
else
|
|
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 */
|
|
if (sound_files[n])
|
|
free (sound_files[n]);
|
|
sound_files[n] = 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);
|
|
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"),
|
|
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 ())
|
|
{
|
|
setup_add_page (cata[9], book, setup_create_page (alert_settings_unity));
|
|
}
|
|
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));
|
|
|
|
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)
|
|
{
|
|
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 ();
|
|
}
|
|
|
|
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;
|
|
|
|
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_tab_dots))
|
|
do_layout = TRUE;
|
|
if (DIFF (hex_gui_tab_layout))
|
|
do_layout = 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);
|
|
|
|
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);
|
|
|
|
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);
|
|
}
|