aboutsummaryrefslogtreecommitdiffstats
path: root/shell/Evolution-ShellComponent.idl
Commit message (Expand)AuthorAgeFilesLines
* Pass NULL for @has_checkbox_func and @has_checkbox_func_data.Ettore Perazzoli2002-05-211-2/+1
* Check with e_shell_prepare_for_quit() before quitting.Ettore Perazzoli2002-05-161-0/+8
* Add some comments to document the interface.Ettore Perazzoli2002-05-161-13/+45
* Add a "view_info" argument.Dan Winship2002-04-091-1/+2
* Move the ComponentActionsPlaceholder into the Actions menu, instead of theEttore Perazzoli2002-03-211-0/+2
* [Implement an Outlook-style "New" dropdown button. It is basicallyEttore Perazzoli2002-01-241-0/+1
* New member `is_interactive' in `EShellPrivate'. (init): Init to %FALSE.Ettore Perazzoli2001-10-231-0/+2
* If setting the owner fails, print the a warning message out. Then restartEttore Perazzoli2001-10-061-2/+2
* Studlycapsize all the IDs that are not. Likewise. Updated accordingly.Ettore Perazzoli2001-08-281-8/+8
* Added display_name and description to the type.Ettore Perazzoli2001-08-201-0/+3
* Updated to display the parent folder's URI and its type as well.Ettore Perazzoli2001-08-101-1/+3
* Add the menu items using the EShellUserCreatableItemsHandler.Ettore Perazzoli2001-08-101-0/+12
* Add an extra @type arg to the xferFolder and removeFolder methods inEttore Perazzoli2001-07-231-1/+3
* Add a `user_creatable' property to folder types and make componentsEttore Perazzoli2001-07-221-0/+2
* Make the shell to be able to display URIs that the user specified onEttore Perazzoli2001-06-281-1/+3
* Add method `ShellComponent::handleExternalURI' and anEttore Perazzoli2001-06-281-2/+7
* add a "debug" method to tell a component to output debugging messages to aDan Winship2001-03-311-0/+2
* Changed the EStorageSet API to have a single `xfer()' method insteadEttore Perazzoli2001-03-221-1/+1
* s/helixcode\.com/ximian.com/ and update all the copyrights in the IDLEttore Perazzoli2001-03-211-2/+2
* Initial step for refactoring of the DnD API for ShellComponents.Ettore Perazzoli2001-03-191-8/+0
* Added a ShellComponent method to retrieve the selection for drag andEttore Perazzoli2001-03-081-1/+9
* Added the ability for components to specify drag/drop types, andEttore Perazzoli2001-03-081-2/+2
* Added `accepted_dnd_mime_types' and `exported_dnd_mime_types' membersEttore Perazzoli2001-02-271-0/+3
* Add an ::asyncCopyFolder method to the ShellComponent interface. MoveEttore Perazzoli2001-01-131-0/+6
* Rename `::addFolderAsync' into `::createFolderAsync'.Ettore Perazzoli2001-01-131-3/+3
* A very, long, very tedious IDL API rename and re-scoping;Michael Meeks2000-11-111-13/+16
* s/Bonobo_UIHandler/Bonobo_UIContainer/Michael Meeks2000-09-211-1/+1
* add "in string evolution_homedir" to ShellComponent::set_owner'sDan Winship2000-08-111-1/+1
* Add interfaces to populate a folder's context menu. (Untested.)Ettore Perazzoli2000-07-211-0/+4
* Initial support for creating/removing folders. This implies some newEttore Perazzoli2000-06-101-3/+37
* Make quit work again by implementing a CORBA method to unregisterEttore Perazzoli2000-05-311-0/+4
* Reorganized the shell to allow dynamic registration of storages andEttore Perazzoli2000-05-251-1/+19
* Get rid of the old `evolution-service-repository' cruft and startEttore Perazzoli2000-05-231-0/+17
' href='#n426'>426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712
/*
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2 of the License, or (at your option) version 3.
 *
 * 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with the program; if not, see <http://www.gnu.org/licenses/>
 *
 *
 * Authors:
 *      Ettore Perazzoli <ettore@ximian.com>
 *
 * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
 *
 */

#include <config.h>

#include <gtk/gtk.h>
#include <glib/gi18n.h>
#include <glib/gstdio.h>

#ifdef G_OS_UNIX
#include <glib-unix.h>
#endif

#ifdef WITH_CONTACT_MAPS
#include <clutter-gtk/clutter-gtk.h>
#endif

#ifdef G_OS_WIN32
#define WIN32_LEAN_AND_MEAN
#ifdef DATADIR
#undef DATADIR
#endif
#define _WIN32_WINNT 0x0601
#include <windows.h>
#include <conio.h>
#include <io.h>
#ifndef PROCESS_DEP_ENABLE
#define PROCESS_DEP_ENABLE 0x00000001
#endif
#ifndef PROCESS_DEP_DISABLE_ATL_THUNK_EMULATION
#define PROCESS_DEP_DISABLE_ATL_THUNK_EMULATION 0x00000002
#endif
#endif

#include <libxml/xmlmemory.h>
#include <libxml/parser.h>
#include <libxml/tree.h>

#include <webkit/webkit.h>

#include "e-shell.h"
#include "e-shell-migrate.h"
#include "e-shell-meego.h"
#include "es-event.h"

#ifdef G_OS_WIN32
#include "e-util/e-win32-defaults.h"
#endif

#include <fcntl.h>
#include <signal.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>

#ifdef HAVE_ICAL_UNKNOWN_TOKEN_HANDLING
#include <libical/ical.h>
#endif

#define APPLICATION_ID "org.gnome.Evolution"

/* STABLE_VERSION is only defined for development versions. */
#ifdef STABLE_VERSION
#define DEVELOPMENT 1
#endif

/* Command-line options.  */
#ifdef G_OS_WIN32
static gboolean register_handlers = FALSE;
static gboolean reinstall = FALSE;
static gboolean show_icons = FALSE;
static gboolean hide_icons = FALSE;
static gboolean unregister_handlers = FALSE;
#endif /* G_OS_WIN32 */
static gboolean express_mode = FALSE;
static gboolean force_online = FALSE;
static gboolean start_online = FALSE;
static gboolean start_offline = FALSE;
static gboolean setup_only = FALSE;
static gboolean force_shutdown = FALSE;
static gboolean disable_eplugin = FALSE;
static gboolean disable_preview = FALSE;
static gboolean import_uris = FALSE;
static gboolean quit = FALSE;

static gchar *geometry = NULL;
static gchar *requested_view = NULL;
static gchar **remaining_args;

/* Forward declarations */
void e_convert_local_mail (EShell *shell);

static void
categories_icon_theme_hack (void)
{
    GtkIconTheme *icon_theme;
    const gchar *category_name;
    const gchar *filename;
    gchar *dirname;

    /* XXX Allow the category icons to be referenced as named
     *     icons, since GtkAction does not support GdkPixbufs. */

    /* Get the icon file for some default category.  Doesn't matter
     * which, so long as it has an icon.  We're just interested in
     * the directory components. */
    category_name = _("Birthday");
    filename = e_categories_get_icon_file_for (category_name);
    g_return_if_fail (filename != NULL && *filename != '\0');

    /* Extract the directory components. */
    dirname = g_path_get_dirname (filename);

    /* Add it to the icon theme's search path.  This relies on
     * GtkIconTheme's legacy feature of using image files found
     * directly in the search path. */
    icon_theme = gtk_icon_theme_get_default ();
    gtk_icon_theme_append_search_path (icon_theme, dirname);

    g_free (dirname);
}

#ifdef DEVELOPMENT

/* Warning dialog to scare people off a little bit.  */

static gboolean
show_development_warning (void)
{
    GtkWidget *action_area;
    GtkWidget *content_area;
    GtkWidget *vbox;
    GtkWidget *label;
    GtkWidget *warning_dialog;
    GtkWidget *checkbox;
    GtkWidget *alignment;
    gboolean skip;
    gchar *text;

    warning_dialog = gtk_dialog_new ();
    gtk_window_set_title (
        GTK_WINDOW (warning_dialog), "Evolution " VERSION);
    gtk_window_set_modal (
        GTK_WINDOW (warning_dialog), TRUE);
    gtk_dialog_add_button (
        GTK_DIALOG (warning_dialog),
        GTK_STOCK_OK, GTK_RESPONSE_OK);

    action_area = gtk_dialog_get_action_area (GTK_DIALOG (warning_dialog));
    content_area = gtk_dialog_get_content_area (GTK_DIALOG (warning_dialog));

    gtk_container_set_border_width (GTK_CONTAINER (action_area), 12);
    gtk_container_set_border_width (GTK_CONTAINER (content_area), 0);

    vbox = gtk_vbox_new (FALSE, 12);
    gtk_container_set_border_width (GTK_CONTAINER (vbox), 12);
    gtk_box_pack_start (GTK_BOX (content_area), vbox, TRUE, TRUE, 0);

    text = g_strdup_printf (
        /* xgettext:no-c-format */
        /* Preview/Alpha/Beta version warning message */
        _("Hi.  Thanks for taking the time to download this preview release\n"
        "of the Evolution groupware suite.\n"
        "\n"
        "This version of Evolution is not yet complete. It is getting close,\n"
        "but some features are either unfinished or do not work properly.\n"
        "\n"
        "If you want a stable version of Evolution, we urge you to uninstall\n"
        "this version, and install version %s instead.\n"
        "\n"
        "If you find bugs, please report them to us at bugzilla.gnome.org.\n"
        "This product comes with no warranty and is not intended for\n"
        "individuals prone to violent fits of anger.\n"
        "\n"
        "We hope that you enjoy the results of our hard work, and we\n"
        "eagerly await your contributions!\n"),
        STABLE_VERSION);
    label = gtk_label_new (text);
    g_free (text);

    gtk_label_set_justify (GTK_LABEL (label), GTK_JUSTIFY_LEFT);
    gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.0);

    gtk_box_pack_start (GTK_BOX (vbox), label, TRUE, TRUE, 0);

    label = gtk_label_new (_("Thanks\nThe Evolution Team\n"));
    gtk_label_set_justify (GTK_LABEL (label), GTK_JUSTIFY_RIGHT);
    gtk_misc_set_alignment (GTK_MISC (label), 1, .5);

    gtk_box_pack_start (GTK_BOX (vbox), label, TRUE, TRUE, 0);

    checkbox = gtk_check_button_new_with_label (_("Do not tell me again"));

    alignment = gtk_alignment_new (0.0, 0.0, 0.0, 0.0);

    gtk_container_add (GTK_CONTAINER (alignment), checkbox);
    gtk_box_pack_start (GTK_BOX (vbox), alignment, TRUE, TRUE, 0);

    gtk_widget_show_all (warning_dialog);

    gtk_dialog_run (GTK_DIALOG (warning_dialog));

    skip = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (checkbox));

    gtk_widget_destroy (warning_dialog);

    return skip;
}

#endif /* DEVELOPMENT */

/* This is for doing stuff that requires the GTK+ loop to be running already.  */

static gboolean
idle_cb (const gchar * const *uris)
{
    EShell *shell;

    shell = e_shell_get_default ();

    /* These calls do the right thing when another Evolution
     * process is running. */
    if (uris != NULL && *uris != NULL) {
        if (e_shell_handle_uris (shell, uris, import_uris) == 0)
            gtk_main_quit ();
    } else {
        if (express_mode && requested_view == NULL)
            requested_view = (gchar *) "mail";
        e_shell_create_shell_window (shell, requested_view);
    }

    /* If another Evolution process is running, we're done. */
    if (g_application_get_is_remote (G_APPLICATION (shell)))
        gtk_main_quit ();

    return FALSE;
}

#ifdef G_OS_UNIX
static gboolean
handle_term_signal (gpointer data)
{
    EShell *shell;

    g_print ("Received terminate signal...\n");

    shell = e_shell_get_default ();

    if (shell != NULL)
        e_shell_quit (shell, E_SHELL_QUIT_OPTION);

    return FALSE;
}
#endif

G_GNUC_NORETURN static gboolean
option_version_cb (const gchar *option_name,
                   const gchar *option_value,
                   gpointer data,
                   GError **error)
{
    g_print ("%s\n", PACKAGE_STRING);

    exit (0);
}

static GOptionEntry entries[] = {
#ifdef G_OS_WIN32
    { "register-handlers", '\0', G_OPTION_FLAG_HIDDEN,
      G_OPTION_ARG_NONE, &register_handlers, NULL, NULL },
    { "reinstall", '\0', G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_NONE, &reinstall,
      NULL, NULL },
    { "show-icons", '\0', G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_NONE, &show_icons,
      NULL, NULL },
    { "hide-icons", '\0', G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_NONE, &hide_icons,
      NULL, NULL },
    { "unregister-handlers", '\0', G_OPTION_FLAG_HIDDEN,
      G_OPTION_ARG_NONE, &unregister_handlers, NULL, NULL },
#endif /* G_OS_WIN32 */
    { "component", 'c', 0, G_OPTION_ARG_STRING, &requested_view,
    /* Translators: Do NOT translate the five component
     * names, they MUST remain in English! */
      N_("Start Evolution showing the specified component. "
         "Available options are 'mail', 'calendar', 'contacts', "
         "'tasks', and 'memos'"), NULL },
    { "geometry", 'g', 0, G_OPTION_ARG_STRING, &geometry,
      N_("Apply the given geometry to the main window"), "GEOMETRY" },
    { "offline", '\0', 0, G_OPTION_ARG_NONE, &start_offline,
      N_("Start in offline mode"), NULL },
    { "online", '\0', 0, G_OPTION_ARG_NONE, &start_online,
      N_("Start in online mode"), NULL },
    { "force-online", '\0', 0, G_OPTION_ARG_NONE, &force_online,
      N_("Ignore network availability"), NULL },
    { "express", '\0', 0, G_OPTION_ARG_NONE, &express_mode,
      N_("Start in \"express\" mode"), NULL },
#ifdef KILL_PROCESS_CMD
    { "force-shutdown", '\0', 0, G_OPTION_ARG_NONE, &force_shutdown,
      N_("Forcibly shut down Evolution"), NULL },
#endif
    { "disable-eplugin", '\0', 0, G_OPTION_ARG_NONE, &disable_eplugin,
      N_("Disable loading of any plugins."), NULL },
    { "disable-preview", '\0', 0, G_OPTION_ARG_NONE, &disable_preview,
      N_("Disable preview pane of Mail, Contacts and Tasks."), NULL },
    { "setup-only", '\0', G_OPTION_FLAG_HIDDEN, G_OPTION_ARG_NONE,
      &setup_only, NULL, NULL },
    { "import", 'i', 0, G_OPTION_ARG_NONE, &import_uris,
      N_("Import URIs or filenames given as rest of arguments."), NULL },
    { "quit", 'q', 0, G_OPTION_ARG_NONE, &quit,
      N_("Request a running Evolution process to quit"), NULL },
    { "version", 'v', G_OPTION_FLAG_HIDDEN | G_OPTION_FLAG_NO_ARG,
      G_OPTION_ARG_CALLBACK, option_version_cb, NULL, NULL },
    { G_OPTION_REMAINING, 0, 0, G_OPTION_ARG_STRING_ARRAY,
      &remaining_args, NULL, NULL },
    { NULL }
};

static void G_GNUC_NORETURN
shell_force_shutdown (void)
{
    gchar *filename;

    filename = g_build_filename (EVOLUTION_TOOLSDIR, "killev", NULL);
    execl (filename, "killev", NULL);

    g_assert_not_reached ();
}

static EShell *
create_default_shell (void)
{
    EShell *shell;
    GSettings *settings;
    GApplicationFlags flags;
    gboolean online = TRUE;
    gboolean is_meego = FALSE;
    gboolean small_screen = FALSE;
    GError *error = NULL;

    settings = g_settings_new ("org.gnome.evolution.shell");

    /* Requesting online or offline mode from the command-line
     * should be persistent, just like selecting it in the UI. */

    if (start_online || force_online) {
        online = TRUE;
        g_settings_set_boolean (settings, "start-offline", FALSE);
    } else if (start_offline) {
        online = FALSE;
        g_settings_set_boolean (settings, "start-offline", TRUE);
    } else {
        gboolean value;

        value = g_settings_get_boolean (settings, "start-offline");
        if (error == NULL)
            online = !value;
    }

    if (error != NULL) {
        g_warning ("%s", error->message);
        g_clear_error (&error);
    }

    /* Determine whether to run Evolution in "express" mode. */

    if (!express_mode)
        express_mode = g_settings_get_boolean (settings, "express-mode");

    if (!express_mode)
        e_shell_detect_meego (&is_meego, &small_screen);

    if (is_meego)
        express_mode = TRUE;

    if (error != NULL) {
        g_warning ("%s", error->message);
        g_clear_error (&error);
    }

    flags = G_APPLICATION_HANDLES_OPEN |
        G_APPLICATION_HANDLES_COMMAND_LINE;

    shell = g_initable_new (
        E_TYPE_SHELL, NULL, &error,
        "application-id", APPLICATION_ID,
        "flags", flags,
        "geometry", geometry,
        "module-directory", EVOLUTION_MODULEDIR,
        "meego-mode", is_meego,
        "express-mode", express_mode,
        "small-screen-mode", small_screen,
        "online", online,
        "register-session", TRUE,
        NULL);

    /* Failure to register is fatal. */
    if (error != NULL) {
        e_notice (
            NULL, GTK_MESSAGE_ERROR,
            _("Cannot start Evolution.  Another Evolution "
            "instance may be unresponsive. System error: %s"),
            error->message);
        g_clear_error (&error);
    }

    if (force_online && shell)
        e_shell_lock_network_available (shell);

    g_object_unref (settings);

    return shell;
}

gint
main (gint argc,
      gchar **argv)
{
    EShell *shell;
    GSettings *settings;
#ifdef DEVELOPMENT
    gboolean skip_warning_dialog;
#endif
    GError *error = NULL;

#ifdef G_OS_WIN32
    gchar *path;

    /* Reduce risks */
    {
        typedef BOOL (WINAPI *t_SetDllDirectoryA) (LPCSTR lpPathName);
        t_SetDllDirectoryA p_SetDllDirectoryA;

        p_SetDllDirectoryA = GetProcAddress (
            GetModuleHandle ("kernel32.dll"),
            "SetDllDirectoryA");
        if (p_SetDllDirectoryA)
            (*p_SetDllDirectoryA) ("");
    }
#ifndef _WIN64
    {
        typedef BOOL (WINAPI *t_SetProcessDEPPolicy) (DWORD dwFlags);
        t_SetProcessDEPPolicy p_SetProcessDEPPolicy;

        p_SetProcessDEPPolicy = GetProcAddress (
            GetModuleHandle ("kernel32.dll"),
            "SetProcessDEPPolicy");
        if (p_SetProcessDEPPolicy)
            (*p_SetProcessDEPPolicy) (
                PROCESS_DEP_ENABLE |
                PROCESS_DEP_DISABLE_ATL_THUNK_EMULATION);
    }
#endif

    if (fileno (stdout) != -1 && _get_osfhandle (fileno (stdout)) != -1) {
        /* stdout is fine, presumably redirected to a file or pipe */
    } else {
        typedef BOOL (* WINAPI AttachConsole_t) (DWORD);

        AttachConsole_t p_AttachConsole =
            (AttachConsole_t) GetProcAddress (
            GetModuleHandle ("kernel32.dll"), "AttachConsole");

        if (p_AttachConsole && p_AttachConsole (ATTACH_PARENT_PROCESS)) {
            freopen ("CONOUT$", "w", stdout);
            dup2 (fileno (stdout), 1);
            freopen ("CONOUT$", "w", stderr);
            dup2 (fileno (stderr), 2);
        }
    }
#endif

    /* Make ElectricFence work.  */
    free (malloc (10));

    bindtextdomain (GETTEXT_PACKAGE, EVOLUTION_LOCALEDIR);
    bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
    textdomain (GETTEXT_PACKAGE);

    g_type_init ();

    /* do not require Gtk+ for --force-shutdown */
    if (argc == 2 && argv[1] && g_str_equal (argv[1], "--force-shutdown")) {
        shell_force_shutdown ();

        return 0;
    }

    /* The contact maps feature uses clutter-gtk. */
#ifdef WITH_CONTACT_MAPS
    /* XXX This function is declared in gtk-clutter-util.h with an
     *     unnecessary G_GNUC_WARN_UNUSED_RESULT attribute.  But we
     *     don't need the returned error code because we're checking
     *     the GError directly.  Just ignore this warning. */
    gtk_clutter_init_with_args (
        &argc, &argv,
        _("- The Evolution PIM and Email Client"),
        entries, (gchar *) GETTEXT_PACKAGE, &error);
#else
    gtk_init_with_args (
        &argc, &argv,
        _("- The Evolution PIM and Email Client"),
        entries, (gchar *) GETTEXT_PACKAGE, &error);
#endif /* WITH_CONTACT_MAPS */

    if (error != NULL) {
        g_printerr ("%s\n", error->message);
        g_error_free (error);
        exit (1);
    }

#ifdef HAVE_ICAL_UNKNOWN_TOKEN_HANDLING
    ical_set_unknown_token_handling_setting (ICAL_DISCARD_TOKEN);
#endif

    e_gdbus_templates_init_main_thread ();

#ifdef G_OS_WIN32
    path = g_build_path (";", _e_get_bindir (), g_getenv ("PATH"), NULL);

    if (!g_setenv ("PATH", path, TRUE))
        g_warning ("Could not set PATH for Evolution and its child processes");

    g_free (path);

    if (register_handlers || reinstall || show_icons) {
        _e_win32_register_mailer ();
        _e_win32_register_addressbook ();
    }

    if (register_handlers)
        exit (0);

    if (reinstall) {
        _e_win32_set_default_mailer ();
        exit (0);
    }

    if (show_icons) {
        _e_win32_set_default_mailer ();
        exit (0);
    }

    if (hide_icons) {
        _e_win32_unset_default_mailer ();
        exit (0);
    }

    if (unregister_handlers) {
        _e_win32_unregister_mailer ();
        _e_win32_unregister_addressbook ();
        exit (0);
    }

    if (strcmp (gettext (""), "") == 0) {
        /* No message catalog installed for the current locale
         * language, so don't bother with the localisations
         * provided by other things then either. Reset thread
         * locale to "en-US" and C library locale to "C". */
        SetThreadLocale (
            MAKELCID (MAKELANGID (LANG_ENGLISH, SUBLANG_ENGLISH_US),
            SORT_DEFAULT));
        setlocale (LC_ALL, "C");
    }
#endif

    if (start_online && start_offline) {
        g_printerr (
            _("%s: --online and --offline cannot be used "
            "together.\n  Run '%s --help' for more "
            "information.\n"), argv[0], argv[0]);
        exit (1);
    } else if (force_online && start_offline) {
        g_printerr (
            _("%s: --force-online and --offline cannot be used "
            "together.\n  Run '%s --help' for more "
            "information.\n"), argv[0], argv[0]);
        exit (1);
    }

    if (force_shutdown)
        shell_force_shutdown ();

    if (disable_preview) {
        settings = g_settings_new ("org.gnome.evolution.mail");
        g_settings_set_boolean (settings, "safe-list", TRUE);
        g_object_unref (settings);

        settings = g_settings_new ("org.gnome.evolution.addressbook");
        g_settings_set_boolean (settings, "show-preview", FALSE);
        g_object_unref (settings);

        settings = g_settings_new ("org.gnome.evolution.calendar");
        g_settings_set_boolean (settings, "show-memo-preview", FALSE);
        g_settings_set_boolean (settings, "show-task-preview", FALSE);
        g_object_unref (settings);
    }

#ifdef G_OS_UNIX
    g_unix_signal_add_full (
        G_PRIORITY_DEFAULT, SIGTERM,
        handle_term_signal, NULL, NULL);
#endif

    e_passwords_init ();

    gtk_window_set_default_icon_name ("evolution");

    if (setup_only)
        exit (0);

    categories_icon_theme_hack ();
    gtk_accel_map_load (e_get_accels_filename ());

#ifdef DEVELOPMENT
    settings = g_settings_new ("org.gnome.evolution.shell");
    skip_warning_dialog = g_settings_get_boolean (
        settings, "skip-warning-dialog");

    if (!skip_warning_dialog && !getenv ("EVOLVE_ME_HARDER"))
        g_settings_set_boolean (
            settings, "skip-warning-dialog",
            show_development_warning ());

    g_object_unref (settings);
#endif

    /* Workaround https://bugzilla.gnome.org/show_bug.cgi?id=683548 */
    if (!quit)
        g_type_ensure (WEBKIT_TYPE_WEB_VIEW);

    shell = create_default_shell ();
    if (!shell)
        return 1;

    if (quit) {
        e_shell_quit (shell, E_SHELL_QUIT_OPTION);
        goto exit;
    }

    /* This routine converts the local mail store from mbox format to
     * Maildir format as needed.  The reason the code is here and not
     * in the mail module is because we inform the user at startup of
     * the impending mail conversion by displaying a popup dialog and
     * waiting for confirmation before proceeding.
     *
     * This has to be done before we load modules because some of the
     * EShellBackends immediately add GMainContext sources that would
     * otherwise get dispatched during gtk_dialog_run(), and we don't
     * want them dispatched until after the conversion is complete. */
    e_convert_local_mail (shell);

    e_shell_load_modules (shell);

    if (!disable_eplugin) {
        /* Register built-in plugin hook types. */
        es_event_hook_get_type ();
        g_type_ensure (E_TYPE_IMPORT_HOOK);
        g_type_ensure (E_TYPE_PLUGIN_UI_HOOK);

        /* All EPlugin and EPluginHook subclasses should be
         * registered in GType now, so load plugins now. */
        e_plugin_load_plugins ();
    }

    if (requested_view)
        e_shell_set_startup_view (shell, requested_view);
    else if (express_mode)
        e_shell_set_startup_view (shell, "mail");

    /* Attempt migration -after- loading all modules and plugins,
     * as both shell backends and certain plugins hook into this. */
    e_shell_migrate_attempt (shell);

    /* This is an alternative to the "upgrade.done" EEvent. */
    e_shell_event (shell, "ready-to-start", NULL);

    g_idle_add ((GSourceFunc) idle_cb, remaining_args);

    gtk_main ();

exit:
    /* Drop what should be the last reference to the shell.
     * That will cause e_shell_get_default() to henceforth
     * return NULL.  Use that to check for reference leaks. */
    g_object_unref (shell);

    if (e_shell_get_default () != NULL)
        g_warning ("Shell not finalized on exit");

    gtk_accel_map_save (e_get_accels_filename ());

    return 0;
}