mirror of
https://github.com/moparisthebest/hexchat
synced 2024-08-13 16:53:48 -04:00
449553ffb0
Xtext's transparency barely worked on windows, didn't work on any modern linux wm and used fake transparency. This uses gtk's built in window opacity that works on more systems and is real transparency. Text area only transparency may return with a transition to cairo, if it works on Windows.
2351 lines
70 KiB
C
2351 lines
70 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
|
|
#ifdef USE_GTKSPELL
|
|
#include <gtkspell/gtkspell.h>
|
|
#endif
|
|
#ifdef USE_LIBSEXY
|
|
#include "sexy-spell-entry.h"
|
|
#endif
|
|
|
|
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_("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_("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},
|
|
#ifdef HAVE_ISO_CODES /* Defined with static spelling */
|
|
{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 \"share\\myspell\\dicts\").\nSeparate multiple entries with commas.")},
|
|
#else
|
|
{ST_LABEL, N_("Use language codes. Separate multiple entries with commas.")},
|
|
#endif
|
|
#else
|
|
#if defined(USE_GTKSPELL) || defined(USE_LIBSEXY)
|
|
{ST_TOGGLE, N_("Spell checking"), P_OFFINTNL(hex_gui_input_spell),0,0,0},
|
|
#endif
|
|
#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},
|
|
|
|
#if 0 /* obsolete */
|
|
{ST_HEADER, N_("Input Box Codes"),0,0,0},
|
|
{ST_TOGGLE, N_("Interpret %nnn as an ASCII value"), P_OFFINTNL(hex_input_perc_ascii),0,0,0},
|
|
{ST_TOGGLE, N_("Interpret %C, %B as Color, Bold etc"), P_OFFINTNL(hex_input_perc_color),0,0,0},
|
|
#endif
|
|
|
|
{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_("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},
|
|
#ifndef WIN32
|
|
{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},
|
|
{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},
|
|
#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_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)
|
|
{
|
|
add_tip (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)
|
|
add_tip (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)
|
|
add_tip (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;
|
|
}
|
|
|
|
#if 0
|
|
static void
|
|
setup_create_toggle (GtkWidget *box, 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)
|
|
add_tip (wid, _(set->tooltip));
|
|
gtk_box_pack_start (GTK_BOX (box), wid, 0, 0, 0);
|
|
}
|
|
#endif
|
|
|
|
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)
|
|
add_tip (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)
|
|
add_tip (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_new_text ();
|
|
|
|
for (i = 0; text[i]; i++)
|
|
gtk_combo_box_append_text (GTK_COMBO_BOX (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;
|
|
|
|
dialog = (GtkFontSelectionDialog *) gtk_font_selection_dialog_new (_("Select font"));
|
|
font_dialog = (GtkWidget *)dialog; /* global var */
|
|
|
|
sel = (GtkFontSelection *) dialog->fontsel;
|
|
|
|
if (gtk_entry_get_text (GTK_ENTRY (entry))[0])
|
|
gtk_font_selection_set_font_name (sel, gtk_entry_get_text (GTK_ENTRY (entry)));
|
|
|
|
g_object_set_data (G_OBJECT (dialog->ok_button), "e", entry);
|
|
|
|
g_signal_connect (G_OBJECT (dialog), "destroy",
|
|
G_CALLBACK (setup_fontsel_destroy), dialog);
|
|
g_signal_connect (G_OBJECT (dialog->ok_button), "clicked",
|
|
G_CALLBACK (setup_fontsel_cb), dialog);
|
|
g_signal_connect (G_OBJECT (dialog->cancel_button), "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)
|
|
add_tip (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 GtkWidget *
|
|
setup_create_frame (GtkWidget **left, GtkWidget *box)
|
|
{
|
|
GtkWidget *tab, *hbox, *inbox = box;
|
|
|
|
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);
|
|
gtk_container_add (GTK_CONTAINER (inbox), tab);
|
|
|
|
hbox = gtk_hbox_new (FALSE, 0);
|
|
gtk_container_add (GTK_CONTAINER (inbox), hbox);
|
|
|
|
*left = gtk_vbox_new (FALSE, 0);
|
|
gtk_box_pack_start (GTK_BOX (hbox), *left, 0, 0, 0);
|
|
|
|
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, *box, *left;
|
|
GtkWidget *wid = NULL, *parentwid = NULL;
|
|
|
|
box = gtk_vbox_new (FALSE, 1);
|
|
gtk_container_set_border_width (GTK_CONTAINER (box), 6);
|
|
|
|
tab = setup_create_frame (&left, box);
|
|
|
|
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 0
|
|
if (set == general_settings)
|
|
{
|
|
setup_create_id_menu (tab, _("Mark identified users with:"),
|
|
row, setup_prefs.hex_irc_id_ytext);
|
|
setup_create_id_menu (tab, _("Mark not-identified users with:"),
|
|
row + 1, setup_prefs.hex_irc_id_ntext);
|
|
}
|
|
#endif
|
|
|
|
if (set == logging_settings)
|
|
{
|
|
GtkWidget *but = gtk_button_new_with_label (_("Open Data Folder"));
|
|
gtk_box_pack_start (GTK_BOX (left), but, 0, 0, 0);
|
|
g_signal_connect (G_OBJECT (but), "clicked",
|
|
G_CALLBACK (open_data_cb), 0);
|
|
}
|
|
|
|
return box;
|
|
}
|
|
|
|
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 (cdialog->colorsel), 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;
|
|
GtkColorSelectionDialog *cdialog;
|
|
GdkColor *color;
|
|
|
|
color = &colors[GPOINTER_TO_INT (userdata)];
|
|
|
|
dialog = gtk_color_selection_dialog_new (_("Select color"));
|
|
cdialog = GTK_COLOR_SELECTION_DIALOG (dialog);
|
|
|
|
gtk_widget_hide (cdialog->help_button);
|
|
g_signal_connect (G_OBJECT (cdialog->ok_button), "clicked",
|
|
G_CALLBACK (setup_color_ok_cb), dialog);
|
|
g_signal_connect (G_OBJECT (cdialog->cancel_button), "clicked",
|
|
G_CALLBACK (gtkutil_destroy), dialog);
|
|
g_object_set_data (G_OBJECT (cdialog->ok_button), "c", color);
|
|
g_object_set_data (G_OBJECT (cdialog->ok_button), "b", button);
|
|
gtk_widget_set_sensitive (cdialog->help_button, FALSE);
|
|
gtk_color_selection_set_current_color (GTK_COLOR_SELECTION (cdialog->colorsel), color);
|
|
gtk_widget_show (dialog);
|
|
}
|
|
|
|
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);
|
|
#ifdef HAVE_ISO_CODES /* Defined with static spelling */
|
|
setup_create_other_colorR (_("Spell checker:"), COL_SPELL, 11, tab);
|
|
#endif
|
|
|
|
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 *oframe, *frame, *label, *vvbox;
|
|
char buf[128];
|
|
|
|
/* frame for whole page */
|
|
oframe = gtk_frame_new (NULL);
|
|
gtk_frame_set_shadow_type (GTK_FRAME (oframe), GTK_SHADOW_IN);
|
|
|
|
vvbox = gtk_vbox_new (FALSE, 0);
|
|
gtk_container_add (GTK_CONTAINER (oframe), vvbox);
|
|
|
|
/* border for the label */
|
|
frame = gtk_frame_new (NULL);
|
|
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_OUT);
|
|
gtk_box_pack_start (GTK_BOX (vvbox), frame, FALSE, TRUE, 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_container_add (GTK_CONTAINER (frame), label);
|
|
|
|
gtk_container_add (GTK_CONTAINER (vvbox), tab);
|
|
|
|
gtk_notebook_append_page (GTK_NOTEBOOK (book), oframe, 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[14], book, setup_create_page (network_settings));
|
|
setup_add_page (cata[15], 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)
|
|
{
|
|
#ifdef USE_GTKSPELL
|
|
GtkSpell *spell;
|
|
#endif
|
|
|
|
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);
|
|
|
|
#ifdef USE_GTKSPELL
|
|
spell = gtkspell_get_from_text_view (GTK_TEXT_VIEW (gui->input_box));
|
|
if (prefs.hex_gui_input_spell)
|
|
{
|
|
if (!spell)
|
|
gtkspell_new_attach (GTK_TEXT_VIEW (gui->input_box), NULL, NULL);
|
|
}
|
|
else
|
|
{
|
|
if (spell)
|
|
gtkspell_detach (spell);
|
|
}
|
|
#endif
|
|
|
|
#ifdef USE_LIBSEXY
|
|
/* 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);
|
|
#endif
|
|
}
|
|
|
|
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];
|
|
time_t rawtime;
|
|
#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);
|
|
*/
|
|
|
|
/* workaround for strftime differences between POSIX and MSVC */
|
|
time (&rawtime);
|
|
|
|
if (!strftime (buffer, sizeof (buffer), prefs.hex_stamp_text_format, localtime (&rawtime)) || !strftime (buffer, sizeof (buffer), prefs.hex_stamp_log_format, localtime (&rawtime)))
|
|
{
|
|
fe_message (_("Invalid time stamp format! See the strftime MSDN article for details."), FE_MSG_ERROR);
|
|
}
|
|
#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)
|
|
{
|
|
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
|
|
}
|
|
|
|
#if 0
|
|
static void
|
|
setup_apply_cb (GtkWidget *but, GtkWidget *win)
|
|
{
|
|
/* setup_prefs -> hexchat */
|
|
setup_apply (&setup_prefs);
|
|
}
|
|
#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));
|
|
|
|
hbox = gtk_hbox_new (FALSE, 0);
|
|
gtk_box_pack_end (GTK_BOX (vbox), hbox, FALSE, FALSE, 0);
|
|
|
|
/* prepare the button box */
|
|
hbbox = gtk_hbutton_box_new ();
|
|
gtk_box_set_spacing (GTK_BOX (hbbox), 4);
|
|
gtk_box_pack_end (GTK_BOX (hbox), hbbox, FALSE, FALSE, 0);
|
|
|
|
/* standard buttons */
|
|
/* GNOME doesn't like apply */
|
|
#if 0
|
|
wid = gtk_button_new_from_stock (GTK_STOCK_APPLY);
|
|
g_signal_connect (G_OBJECT (wid), "clicked",
|
|
G_CALLBACK (setup_apply_cb), win);
|
|
gtk_box_pack_start (GTK_BOX (hbbox), wid, FALSE, FALSE, 0);
|
|
#endif
|
|
|
|
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);
|
|
|
|
wid = gtk_hseparator_new ();
|
|
gtk_box_pack_end (GTK_BOX (vbox), 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);
|
|
}
|