/* fe-common-core.c : irssi Copyright (C) 1999-2000 Timo Sirainen 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 Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "module.h" #include "module-formats.h" #include "args.h" #include "misc.h" #include "levels.h" #include "settings.h" #include "servers.h" #include "channels.h" #include "servers-setup.h" #include "special-vars.h" #include "fe-core-commands.h" #include "fe-queries.h" #include "hilight-text.h" #include "command-history.h" #include "completion.h" #include "keyboard.h" #include "printtext.h" #include "formats.h" #include "themes.h" #include "fe-channels.h" #include "fe-windows.h" #include "window-activity.h" #include "window-items.h" #include "windows-layout.h" #include "fe-recode.h" #include static char *autocon_server; static char *autocon_password; static int autocon_port; static int no_autoconnect; static char *cmdline_nick; static char *cmdline_hostname; void fe_core_log_init(void); void fe_core_log_deinit(void); void fe_exec_init(void); void fe_exec_deinit(void); void fe_expandos_init(void); void fe_expandos_deinit(void); void fe_help_init(void); void fe_help_deinit(void); void fe_ignore_init(void); void fe_ignore_deinit(void); void fe_ignore_messages_init(void); void fe_ignore_messages_deinit(void); void fe_log_init(void); void fe_log_deinit(void); void fe_messages_init(void); void fe_messages_deinit(void); void fe_modules_init(void); void fe_modules_deinit(void); void fe_server_init(void); void fe_server_deinit(void); void fe_settings_init(void); void fe_settings_deinit(void); void window_commands_init(void); void window_commands_deinit(void); static void sig_setup_changed(void); static void sig_connected(SERVER_REC *server) { MODULE_DATA_SET(server, g_new0(MODULE_SERVER_REC, 1)); } static void sig_disconnected(SERVER_REC *server) { void *data = MODULE_DATA(server); g_free(data); MODULE_DATA_UNSET(server); } static void sig_channel_created(CHANNEL_REC *channel) { MODULE_DATA_SET(channel, g_new0(MODULE_CHANNEL_REC, 1)); } static void sig_channel_destroyed(CHANNEL_REC *channel) { void *data = MODULE_DATA(channel); g_free(data); MODULE_DATA_UNSET(channel); } void fe_common_core_register_options(void) { static GOptionEntry options[] = { { "connect", 'c', 0, G_OPTION_ARG_STRING, &autocon_server, "Automatically connect to server/network", "SERVER" }, { "password", 'w', 0, G_OPTION_ARG_STRING, &autocon_password, "Autoconnect password", "PASSWORD" }, { "port", 'p', 0, G_OPTION_ARG_INT, &autocon_port, "Autoconnect port", "PORT" }, { "noconnect", '!', 0, G_OPTION_ARG_NONE, &no_autoconnect, "Disable autoconnecting", NULL }, { "nick", 'n', 0, G_OPTION_ARG_STRING, &cmdline_nick, "Specify nick to use", NULL }, { "hostname", 'h', 0, G_OPTION_ARG_STRING, &cmdline_hostname, "Specify host name to use", NULL }, { NULL } }; autocon_server = NULL; autocon_password = NULL; autocon_port = 0; no_autoconnect = FALSE; cmdline_nick = NULL; cmdline_hostname = NULL; args_register(options); } void fe_common_core_init(void) { const char *str; settings_add_bool("lookandfeel", "timestamps", TRUE); settings_add_level("lookandfeel", "timestamp_level", "ALL"); settings_add_time("lookandfeel", "timestamp_timeout", "0"); settings_add_bool("lookandfeel", "bell_beeps", FALSE); settings_add_level("lookandfeel", "beep_msg_level", ""); settings_add_bool("lookandfeel", "beep_when_window_active", TRUE); settings_add_bool("lookandfeel", "beep_when_away", TRUE); settings_add_bool("lookandfeel", "hide_text_style", FALSE); settings_add_bool("lookandfeel", "hide_colors", FALSE); settings_add_bool("lookandfeel", "hide_server_tags", FALSE); settings_add_bool("lookandfeel", "use_status_window", TRUE); settings_add_bool("lookandfeel", "use_msgs_window", FALSE); g_get_charset(&str); settings_add_str("lookandfeel", "term_charset", str); themes_init(); theme_register(fecommon_core_formats); command_history_init(); completion_init(); keyboard_init(); printtext_init(); formats_init(); #ifndef WIN32 fe_exec_init(); #endif fe_expandos_init(); fe_help_init(); fe_ignore_init(); fe_log_init(); fe_modules_init(); fe_server_init(); fe_settings_init(); windows_init(); window_activity_init(); window_commands_init(); window_items_init(); windows_layout_init(); fe_core_commands_init(); fe_channels_init(); fe_queries_init(); fe_messages_init(); hilight_text_init(); fe_ignore_messages_init(); fe_recode_init(); settings_check(); signal_add_first("server connected", (SIGNAL_FUNC) sig_connected); signal_add_last("server disconnected", (SIGNAL_FUNC) sig_disconnected); signal_add_first("channel created", (SIGNAL_FUNC) sig_channel_created); signal_add_last("channel destroyed", (SIGNAL_FUNC) sig_channel_destroyed); module_register("core", "fe"); } void fe_common_core_deinit(void) { hilight_text_deinit(); command_history_deinit(); completion_deinit(); keyboard_deinit(); printtext_deinit(); formats_deinit(); #ifndef WIN32 fe_exec_deinit(); #endif fe_expandos_deinit(); fe_help_deinit(); fe_ignore_deinit(); fe_log_deinit(); fe_modules_deinit(); fe_server_deinit(); fe_settings_deinit(); windows_deinit(); window_activity_deinit(); window_commands_deinit(); window_items_deinit(); windows_layout_deinit(); fe_core_commands_deinit(); fe_channels_deinit(); fe_queries_deinit(); fe_messages_deinit(); fe_ignore_messages_deinit(); fe_recode_deinit(); theme_unregister(); themes_deinit(); signal_remove("setup changed", (SIGNAL_FUNC) sig_setup_changed); signal_remove("server connected", (SIGNAL_FUNC) sig_connected); signal_remove("server disconnected", (SIGNAL_FUNC) sig_disconnected); signal_remove("channel created", (SIGNAL_FUNC) sig_channel_created); signal_remove("channel destroyed", (SIGNAL_FUNC) sig_channel_destroyed); } void glog_func(const char *log_domain, GLogLevelFlags log_level, const char *message) { const char *reason; switch (log_level) { case G_LOG_LEVEL_WARNING: reason = "warning"; break; case G_LOG_LEVEL_CRITICAL: reason = "critical"; break; default: reason = "error"; break; } if (windows == NULL) fprintf(stderr, "GLib %s: %s\n", reason, message); else { printformat(NULL, NULL, MSGLEVEL_CLIENTERROR, TXT_GLIB_ERROR, reason, message); } } #define MSGS_WINDOW_LEVELS (MSGLEVEL_MSGS|MSGLEVEL_ACTIONS|MSGLEVEL_DCCMSGS) static void create_windows(void) { WINDOW_REC *window; int have_status = settings_get_bool("use_status_window"); window = window_find_name("(status)"); if (have_status) { if (window == NULL) { window = window_create(NULL, TRUE); window_set_refnum(window, 1); window_set_name(window, "(status)"); window_set_level(window, MSGLEVEL_ALL ^ (settings_get_bool("use_msgs_window") ? MSGS_WINDOW_LEVELS : 0)); window_set_immortal(window, TRUE); } } else { if (window != NULL) { window_set_name(window, NULL); window_set_level(window, 0); window_set_immortal(window, FALSE); } } window = window_find_name("(msgs)"); if (settings_get_bool("use_msgs_window")) { if (window == NULL) { window = window_create(NULL, TRUE); window_set_refnum(window, have_status ? 2 : 1); window_set_name(window, "(msgs)"); window_set_level(window, MSGS_WINDOW_LEVELS); window_set_immortal(window, TRUE); } } else { if (window != NULL) { window_set_name(window, NULL); window_set_level(window, 0); window_set_immortal(window, FALSE); } } if (windows == NULL) { /* we have to have at least one window.. */ window = window_create(NULL, TRUE); } } static void autoconnect_servers(void) { GSList *tmp, *chatnets; char *str; if (autocon_server != NULL) { /* connect to specified server */ str = g_strdup_printf(autocon_password == NULL ? "%s %d" : "%s %d %s", autocon_server, autocon_port, autocon_password); signal_emit("command connect", 1, str); g_free(str); return; } if (no_autoconnect) { /* don't autoconnect */ return; } /* connect to autoconnect servers */ chatnets = NULL; for (tmp = setupservers; tmp != NULL; tmp = tmp->next) { SERVER_SETUP_REC *rec = tmp->data; if (rec->autoconnect && (rec->chatnet == NULL || gslist_find_icase_string(chatnets, rec->chatnet) == NULL)) { if (rec->chatnet != NULL) { chatnets = g_slist_append(chatnets, rec->chatnet); str = g_strdup_printf("-network %s %s %d", rec->chatnet, rec->address, rec->port); } else { str = g_strdup_printf("%s %d", rec->address, rec->port); } signal_emit("command connect", 1, str); g_free(str); } } g_slist_free(chatnets); } static void sig_setup_changed(void) { static int firsttime = TRUE; static int status_window = FALSE, msgs_window = FALSE; int changed = FALSE; if (settings_get_bool("use_status_window") != status_window) { status_window = !status_window; changed = TRUE; } if (settings_get_bool("use_msgs_window") != msgs_window) { msgs_window = !msgs_window; changed = TRUE; } if (firsttime) { firsttime = FALSE; changed = TRUE; windows_layout_restore(); if (windows != NULL) return; } if (changed) create_windows(); } static void autorun_startup(void) { char *path; GIOChannel *handle; GString *buf; gsize tpos; /* open ~/.irssi/startup and run all commands in it */ path = g_strdup_printf("%s/startup", get_irssi_dir()); handle = g_io_channel_new_file(path, "r", NULL); g_free(path); if (handle == NULL) { /* file not found */ return; } g_io_channel_set_encoding(handle, NULL, NULL); buf = g_string_sized_new(512); while (g_io_channel_read_line_string(handle, buf, &tpos, NULL) == G_IO_STATUS_NORMAL) { buf->str[tpos] = '\0'; if (buf->str[0] != '#') { eval_special_string(buf->str, "", active_win->active_server, active_win->active); } } g_string_free(buf, TRUE); g_io_channel_unref(handle); } void fe_common_core_finish_init(void) { int setup_changed; signal_emit("irssi init read settings", 0); #ifdef SIGPIPE signal(SIGPIPE, SIG_IGN); #endif setup_changed = FALSE; if (cmdline_nick != NULL && *cmdline_nick != '\0') { /* override nick found from setup */ settings_set_str("nick", cmdline_nick); setup_changed = TRUE; } if (cmdline_hostname != NULL) { /* override host name found from setup */ settings_set_str("hostname", cmdline_hostname); setup_changed = TRUE; } sig_setup_changed(); signal_add_first("setup changed", (SIGNAL_FUNC) sig_setup_changed); /* _after_ windows are created.. */ #if GLIB_CHECK_VERSION(2,6,0) g_log_set_default_handler((GLogFunc) glog_func, NULL); #else g_log_set_handler(G_LOG_DOMAIN, (GLogLevelFlags) (G_LOG_LEVEL_CRITICAL | G_LOG_LEVEL_WARNING), (GLogFunc) glog_func, NULL); g_log_set_handler("GLib", (GLogLevelFlags) (G_LOG_LEVEL_CRITICAL | G_LOG_LEVEL_WARNING), (GLogFunc) glog_func, NULL); /* send glib errors to the same place */ #endif if (setup_changed) signal_emit("setup changed", 0); autorun_startup(); autoconnect_servers(); } gboolean strarray_find_dest(char **array, const TEXT_DEST_REC *dest) { g_return_val_if_fail(array != NULL, FALSE); if (strarray_find(array, dest->target) != -1) return TRUE; if (dest->server_tag != NULL) { char *tagtarget = g_strdup_printf("%s/%s", dest->server_tag, dest->target); int ret = strarray_find(array, tagtarget); g_free(tagtarget); if (ret != -1) return TRUE; } return FALSE; }