merge of '134eb2e58f255d14d53d8b9463921ce7edb9aa36'
[pidgin-git.git] / finch / finch.c
blobd384930f435d1f807c74a3557f8ec44e29fa61e2
1 /**
2 * finch
4 * Finch is the legal property of its developers, whose names are too numerous
5 * to list here. Please refer to the COPYRIGHT file distributed with this
6 * source distribution.
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
22 #include "account.h"
23 #include "conversation.h"
24 #include "core.h"
25 #include "debug.h"
26 #include "eventloop.h"
27 #include "ft.h"
28 #include "log.h"
29 #include "notify.h"
30 #include "prefs.h"
31 #include "prpl.h"
32 #include "pounce.h"
33 #include "savedstatuses.h"
34 #include "sound.h"
35 #include "status.h"
36 #include "util.h"
37 #include "whiteboard.h"
39 #include "gntdebug.h"
40 #include "finch.h"
41 #include "gntprefs.h"
42 #include "gntui.h"
43 #include "gntidle.h"
45 #define _GNU_SOURCE
46 #include <getopt.h>
48 #include "config.h"
50 static void
51 debug_init()
53 finch_debug_init();
54 purple_debug_set_ui_ops(finch_debug_get_ui_ops());
57 static PurpleCoreUiOps core_ops =
59 finch_prefs_init,
60 debug_init,
61 gnt_ui_init,
62 gnt_ui_uninit,
64 /* padding */
65 NULL,
66 NULL,
67 NULL,
68 NULL
71 static PurpleCoreUiOps *
72 gnt_core_get_ui_ops()
74 return &core_ops;
77 /* Anything IO-related is directly copied from gtkpurple's source tree */
79 #define FINCH_READ_COND (G_IO_IN | G_IO_HUP | G_IO_ERR)
80 #define FINCH_WRITE_COND (G_IO_OUT | G_IO_HUP | G_IO_ERR | G_IO_NVAL)
82 typedef struct _PurpleGntIOClosure {
83 PurpleInputFunction function;
84 guint result;
85 gpointer data;
87 } PurpleGntIOClosure;
89 static void purple_gnt_io_destroy(gpointer data)
91 g_free(data);
94 static gboolean purple_gnt_io_invoke(GIOChannel *source, GIOCondition condition, gpointer data)
96 PurpleGntIOClosure *closure = data;
97 PurpleInputCondition purple_cond = 0;
99 if (condition & FINCH_READ_COND)
100 purple_cond |= PURPLE_INPUT_READ;
101 if (condition & FINCH_WRITE_COND)
102 purple_cond |= PURPLE_INPUT_WRITE;
104 #if 0
105 purple_debug(PURPLE_DEBUG_MISC, "gtk_eventloop",
106 "CLOSURE: callback for %d, fd is %d\n",
107 closure->result, g_io_channel_unix_get_fd(source));
108 #endif
110 #ifdef _WIN32
111 if(! purple_cond) {
112 #if DEBUG
113 purple_debug_misc("gnt_eventloop",
114 "CLOSURE received GIOCondition of 0x%x, which does not"
115 " match 0x%x (READ) or 0x%x (WRITE)\n",
116 condition, FINCH_READ_COND, FINCH_WRITE_COND);
117 #endif /* DEBUG */
119 return TRUE;
121 #endif /* _WIN32 */
123 closure->function(closure->data, g_io_channel_unix_get_fd(source),
124 purple_cond);
126 return TRUE;
129 static guint gnt_input_add(gint fd, PurpleInputCondition condition, PurpleInputFunction function,
130 gpointer data)
132 PurpleGntIOClosure *closure = g_new0(PurpleGntIOClosure, 1);
133 GIOChannel *channel;
134 GIOCondition cond = 0;
136 closure->function = function;
137 closure->data = data;
139 if (condition & PURPLE_INPUT_READ)
140 cond |= FINCH_READ_COND;
141 if (condition & PURPLE_INPUT_WRITE)
142 cond |= FINCH_WRITE_COND;
144 channel = g_io_channel_unix_new(fd);
145 closure->result = g_io_add_watch_full(channel, G_PRIORITY_DEFAULT, cond,
146 purple_gnt_io_invoke, closure, purple_gnt_io_destroy);
148 g_io_channel_unref(channel);
149 return closure->result;
152 static PurpleEventLoopUiOps eventloop_ops =
154 g_timeout_add,
155 g_source_remove,
156 gnt_input_add,
157 g_source_remove,
158 NULL, /* input_get_error */
160 /* padding */
161 NULL,
162 NULL,
163 NULL,
164 NULL
167 static PurpleEventLoopUiOps *
168 gnt_eventloop_get_ui_ops(void)
170 return &eventloop_ops;
173 /* This is mostly copied from gtkpurple's source tree */
174 static void
175 show_usage(const char *name, gboolean terse)
177 char *text;
179 if (terse) {
180 text = g_strdup_printf(_("%s. Try `%s -h' for more information.\n"), VERSION, name);
181 } else {
182 text = g_strdup_printf(_("%s\n"
183 "Usage: %s [OPTION]...\n\n"
184 " -c, --config=DIR use DIR for config files\n"
185 " -d, --debug print debugging messages to stdout\n"
186 " -h, --help display this help and exit\n"
187 " -n, --nologin don't automatically login\n"
188 " -v, --version display the current version and exit\n"), VERSION, name);
191 purple_print_utf8_to_console(stdout, text);
192 g_free(text);
195 static int
196 init_libpurple(int argc, char **argv)
198 char *path;
199 int opt;
200 gboolean opt_help = FALSE;
201 gboolean opt_nologin = FALSE;
202 gboolean opt_version = FALSE;
203 char *opt_config_dir_arg = NULL;
204 char *opt_session_arg = NULL;
205 gboolean debug_enabled = FALSE;
207 struct option long_options[] = {
208 {"config", required_argument, NULL, 'c'},
209 {"debug", no_argument, NULL, 'd'},
210 {"help", no_argument, NULL, 'h'},
211 {"nologin", no_argument, NULL, 'n'},
212 {"session", required_argument, NULL, 's'},
213 {"version", no_argument, NULL, 'v'},
214 {0, 0, 0, 0}
217 #ifdef PURPLE_FATAL_ASSERTS
218 /* Make g_return_... functions fatal. */
219 g_log_set_always_fatal(G_LOG_LEVEL_CRITICAL);
220 #endif
222 #ifdef ENABLE_NLS
223 bindtextdomain(PACKAGE, LOCALEDIR);
224 bind_textdomain_codeset(PACKAGE, "UTF-8");
225 textdomain(PACKAGE);
226 #endif
228 #ifdef HAVE_SETLOCALE
229 setlocale(LC_ALL, "");
230 #endif
232 /* scan command-line options */
233 opterr = 1;
234 while ((opt = getopt_long(argc, argv,
235 #ifndef _WIN32
236 "c:dhn::s:v",
237 #else
238 "c:dhn::v",
239 #endif
240 long_options, NULL)) != -1) {
241 switch (opt) {
242 case 'c': /* config dir */
243 g_free(opt_config_dir_arg);
244 opt_config_dir_arg = g_strdup(optarg);
245 break;
246 case 'd': /* debug */
247 debug_enabled = TRUE;
248 break;
249 case 'h': /* help */
250 opt_help = TRUE;
251 break;
252 case 'n': /* no autologin */
253 opt_nologin = TRUE;
254 break;
255 case 's': /* use existing session ID */
256 g_free(opt_session_arg);
257 opt_session_arg = g_strdup(optarg);
258 break;
259 case 'v': /* version */
260 opt_version = TRUE;
261 break;
262 case '?': /* show terse help */
263 default:
264 show_usage(argv[0], TRUE);
265 return 0;
266 break;
270 /* show help message */
271 if (opt_help) {
272 show_usage(argv[0], FALSE);
273 return 0;
275 /* show version message */
276 if (opt_version) {
277 printf("Finch %s\n", VERSION);
278 return 0;
281 /* set a user-specified config directory */
282 if (opt_config_dir_arg != NULL) {
283 purple_util_set_user_dir(opt_config_dir_arg);
284 g_free(opt_config_dir_arg);
288 * We're done piddling around with command line arguments.
289 * Fire up this baby.
292 /* We don't want debug-messages to show up and corrupt the display */
293 purple_debug_set_enabled(debug_enabled);
295 /* If we're using a custom configuration directory, we
296 * do NOT want to migrate, or weird things will happen. */
297 if (opt_config_dir_arg == NULL)
299 if (!purple_core_migrate())
301 char *old = g_strconcat(purple_home_dir(),
302 G_DIR_SEPARATOR_S ".gaim", NULL);
303 char *text = g_strdup_printf(_(
304 "%s encountered errors migrating your settings "
305 "from %s to %s. Please investigate and complete the "
306 "migration by hand. Please report this error at http://developer.pidgin.im"), _("Finch"),
307 old, purple_user_dir());
309 g_free(old);
311 purple_print_utf8_to_console(stderr, text);
312 g_free(text);
314 return 0;
318 purple_core_set_ui_ops(gnt_core_get_ui_ops());
319 purple_eventloop_set_ui_ops(gnt_eventloop_get_ui_ops());
320 purple_idle_set_ui_ops(finch_idle_get_ui_ops());
322 path = g_build_filename(purple_user_dir(), "plugins", NULL);
323 purple_plugins_add_search_path(path);
324 g_free(path);
326 purple_plugins_add_search_path(LIBDIR);
328 if (!purple_core_init(FINCH_UI))
330 fprintf(stderr,
331 "Initialization of the Purple core failed. Dumping core.\n"
332 "Please report this!\n");
333 abort();
336 /* TODO: Move blist loading into purple_blist_init() */
337 purple_set_blist(purple_blist_new());
338 purple_blist_load();
340 /* TODO: Move prefs loading into purple_prefs_init() */
341 purple_prefs_load();
342 purple_prefs_update_old();
343 finch_prefs_update_old();
345 /* load plugins we had when we quit */
346 purple_plugins_load_saved("/finch/plugins/loaded");
348 /* TODO: Move pounces loading into purple_pounces_init() */
349 purple_pounces_load();
351 if (opt_nologin)
353 /* Set all accounts to "offline" */
354 PurpleSavedStatus *saved_status;
356 /* If we've used this type+message before, lookup the transient status */
357 saved_status = purple_savedstatus_find_transient_by_type_and_message(
358 PURPLE_STATUS_OFFLINE, NULL);
360 /* If this type+message is unique then create a new transient saved status */
361 if (saved_status == NULL)
362 saved_status = purple_savedstatus_new(NULL, PURPLE_STATUS_OFFLINE);
364 /* Set the status for each account */
365 purple_savedstatus_activate(saved_status);
367 else
369 /* Everything is good to go--sign on already */
370 if (!purple_prefs_get_bool("/purple/savedstatus/startup_current_status"))
371 purple_savedstatus_activate(purple_savedstatus_get_startup());
372 purple_accounts_restore_current_statuses();
375 return 1;
378 int main(int argc, char **argv)
380 signal(SIGPIPE, SIG_IGN);
382 /* Initialize the libpurple stuff */
383 if (!init_libpurple(argc, argv))
384 return 0;
386 purple_blist_show();
387 gnt_main();
389 #ifdef STANDALONE
390 purple_core_quit();
391 #endif
393 return 0;