/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
/* evolution-test-component.c
*
* Copyright (C) 2001, 2002 Ximian, Inc.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of version 2 of the GNU General Public
* License as published by the Free Software Foundation.
*
* 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., 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*
* Author: Ettore Perazzoli
*/
/* Simple test component for the Evolution shell. */
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include "evolution-shell-component.h"
#include "evolution-activity-client.h"
#include "evolution-config-control.h"
#include "evolution-storage.h"
#include <bonobo-activation/bonobo-activation.h>
#include <bonobo/bonobo-exception.h>
#include <bonobo/bonobo-generic-factory.h>
#include <bonobo/bonobo-main.h>
#include <bonobo/bonobo-widget.h>
#include <libgnomeui/gnome-ui-init.h>
#include <gtk/gtkdialog.h>
#include <gtk/gtkeventbox.h>
#include <gtk/gtklabel.h>
#include <gtk/gtkmain.h>
#include <gtk/gtkstock.h>
#include <gtk/gtkvbox.h>
#include <gdk-pixbuf/gdk-pixbuf.h>
#include <stdlib.h>
#include <string.h>
#define COMPONENT_ID "OAFIID:GNOME_Evolution_TestComponent_ShellComponent"
#define CONFIGURATION_CONTROL_FACTORY_ID "OAFIID:GNOME_Evolution_TestComponent_ConfigurationControlFactory"
#define CONFIGURATION_CONTROL_ID "OAFIID:GNOME_Evolution_TestComponent_ConfigurationControl"
static const EvolutionShellComponentFolderType folder_types[] = {
{ "test", "/usr/share/pixmaps/gnome-money.png", N_("Test"), N_("Test type"), FALSE, NULL, NULL },
{ NULL }
};
static BonoboGenericFactory *configuration_control_factory = NULL;
static EvolutionShellClient *parent_shell = NULL;
static int timeout_id = 0;
/* TEST #1: Configuration Control. */
static BonoboObject *
create_configuration_page (void)
{
GtkWidget *label;
label = gtk_label_new ("This is the configuration page for the test component.");
gtk_widget_show (label);
return BONOBO_OBJECT (evolution_config_control_new (label));
}
static BonoboObject *
configuration_control_factory_fn (BonoboGenericFactory *factory,
const char *id,
void *closure)
{
if (strcmp (id, CONFIGURATION_CONTROL_ID) == 0) {
return create_configuration_page ();
} else {
g_warning ("Unknown ID in configuration control factory -- %s", id);
return NULL;
}
}
static void
register_configuration_control_factory (void)
{
configuration_control_factory = bonobo_generic_factory_new (CONFIGURATION_CONTROL_FACTORY_ID,
configuration_control_factory_fn,
NULL);
if (configuration_control_factory == NULL)
g_warning ("Cannot register configuration control factory!");
}
/* TEST #2: The ::Shortcut interface. */
static void
spit_out_shortcuts (EvolutionShellClient *shell_client)
{
GNOME_Evolution_Shortcuts shortcuts_interface;
GNOME_Evolution_Shortcuts_GroupList *groups;
CORBA_Environment ev;
int i, j;
CORBA_exception_init (&ev);
shortcuts_interface = evolution_shell_client_get_shortcuts_interface (shell_client);
if (CORBA_Object_is_nil (shortcuts_interface, &ev)) {
g_warning ("No ::Shortcut interface on the shell");
CORBA_exception_free (&ev);
return;
}
groups = GNOME_Evolution_Shortcuts__get_groups (shortcuts_interface, &ev);
if (ev._major != CORBA_NO_EXCEPTION) {
g_warning ("Exception getting the groups: %s", BONOBO_EX_REPOID (&ev));
CORBA_exception_free (&ev);
return;
}
for (i = 0; i < groups->_length; i ++) {
GNOME_Evolution_Shortcuts_Group *group;
GNOME_Evolution_Shortcuts_ShortcutList *shortcuts;
group = groups->_buffer + i;
shortcuts = &group->shortcuts;
g_print ("\tGROUP: %s\n", group->name);
for (j = 0; j < shortcuts->_length; j ++) {
GNOME_Evolution_Shortcuts_Shortcut *shortcut;
shortcut = shortcuts->_buffer + j;
g_print ("\t\tName: %s\n", shortcut->name);
g_print ("\t\t\tType: %s\n", shortcut->type);
g_print ("\t\t\tURI: %s\n", shortcut->uri);
}
}
g_print ("** Done\n\n");
CORBA_exception_free (&ev);
}
/* TEST #4: The multiple folder selector. */
static void
dialog_clicked_callback (GtkDialog *dialog,
int button_num,
void *data)
{
GNOME_Evolution_StorageSetView storage_set_view_iface;
CORBA_Environment ev;
GNOME_Evolution_FolderList *folder_list;
if (button_num == 1) {
/* Close. */
gtk_widget_destroy (GTK_WIDGET (dialog));
return;
}
CORBA_exception_init (&ev);
storage_set_view_iface = (GNOME_Evolution_StorageSetView) data;
folder_list = GNOME_Evolution_StorageSetView__get_checkedFolders (storage_set_view_iface, &ev);
if (BONOBO_EX (&ev)) {
g_warning ("Cannot get checkedFolders -- %s", BONOBO_EX_REPOID (&ev));
} else {
int i;
for (i = 0; i < folder_list->_length; i ++) {
#define PRINT(s) g_print ("\t" #s ": %s\n", folder_list->_buffer[i].s);
g_print ("Folder #%d:\n", i + 1);
PRINT (type);
PRINT (description);
PRINT (displayName);
PRINT (physicalUri);
PRINT (evolutionUri);
#undef PRINT
g_print ("\tunreadCount: %d\n", folder_list->_buffer[i].unreadCount);
}
}
CORBA_exception_free (&ev);
}
static void
dialog_weak_notify (void *data,
GObject *where_the_object_was)
{
GNOME_Evolution_StorageSetView storage_set_view_iface;
CORBA_Environment ev;
CORBA_exception_init (&ev);
storage_set_view_iface = (GNOME_Evolution_StorageSetView) data;
Bonobo_Unknown_unref (storage_set_view_iface, &ev);
CORBA_exception_free (&ev);
}
static void
create_new_folder_selector (EvolutionShellComponent *shell_component)
{
EvolutionShellClient *shell_client;
GNOME_Evolution_Shell corba_shell;
GNOME_Evolution_StorageSetView storage_set_view_iface;
GtkWidget *dialog;
GtkWidget *control_widget;
CORBA_Environment ev;
CORBA_exception_init (&ev);
shell_client = evolution_shell_component_get_owner (shell_component);
g_assert (shell_client != NULL);
corba_shell = evolution_shell_client_corba_objref (shell_client);
control_widget = evolution_shell_client_create_storage_set_view (shell_client,
CORBA_OBJECT_NIL,
NULL,
&storage_set_view_iface,
&ev);
if (control_widget == NULL) {
g_warning ("Can't create the StorageSetView control -- %s", BONOBO_EX_REPOID (&ev));
CORBA_exception_free (&ev);
return;
}
dialog = gtk_dialog_new_with_buttons ("Test the Selector here.", NULL,
GTK_DIALOG_MODAL,
GTK_STOCK_APPLY, GTK_RESPONSE_ACCEPT,
GTK_STOCK_CLOSE, GTK_RESPONSE_CLOSE,
NULL);
gtk_window_set_default_size (GTK_WINDOW (dialog), 200, 400);
gtk_container_add (GTK_CONTAINER (GTK_DIALOG (dialog)->vbox), control_widget);
GNOME_Evolution_StorageSetView__set_showCheckboxes (storage_set_view_iface, TRUE, &ev);
if (BONOBO_EX (&ev))
g_warning ("Cannot show checkboxes -- %s", BONOBO_EX_REPOID (&ev));
g_signal_connect (dialog, "clicked",
G_CALLBACK (dialog_clicked_callback), storage_set_view_iface);
/* This is necessary to unref the StorageSetView iface once we are done
with it. */
g_object_weak_ref (G_OBJECT (dialog), dialog_weak_notify, storage_set_view_iface);
gtk_widget_show (control_widget);
gtk_widget_show (dialog);
CORBA_exception_free (&ev);
}
/* TEST #5: Test custom storage. */
static int
shared_folder_discovery_timeout_callback (void *data)
{
GNOME_Evolution_Storage_FolderResult result;
CORBA_Environment ev;
Bonobo_Listener listener;
CORBA_any any;
EvolutionStorage *storage;
storage = EVOLUTION_STORAGE (data);
listener = (Bonobo_Listener) g_object_get_data (G_OBJECT (storage), "listener");
result.result = GNOME_Evolution_Storage_OK;
result.path = "/Shared Folders/The Public Folder";
any._type = TC_GNOME_Evolution_Storage_FolderResult;
any._value = &result;
CORBA_exception_init (&ev);
Bonobo_Listener_event (listener, "result", &any, &ev);
if (BONOBO_EX (&ev))
g_warning ("Cannot report result for shared folder discovery -- %s",
BONOBO_EX_REPOID (&ev));
Bonobo_Unknown_unref (listener, &ev);
CORBA_Object_release (listener, &ev);
CORBA_exception_free (&ev);
g_object_set_data (G_OBJECT (storage), "listener", NULL);
g_object_set_data (G_OBJECT (storage), "timeout_id", NULL);
return FALSE;
}
static void
storage_discover_shared_folder_callback (EvolutionStorage *storage,
Bonobo_Listener listener,
const char *user,
const char *folder_name,
void *data)
{
CORBA_Environment ev;
Bonobo_Listener listener_copy;
CORBA_exception_init (&ev);
listener_copy = CORBA_Object_duplicate (listener, &ev);
Bonobo_Unknown_ref (listener, &ev);
CORBA_exception_free (&ev);
g_print ("Listener copy %p\n", listener_copy);
timeout_id = g_timeout_add (1000, shared_folder_discovery_timeout_callback, storage);
g_object_set_data (G_OBJECT (storage), "listener", listener_copy);
g_object_set_data (G_OBJECT (storage), "timeout_id", GINT_TO_POINTER (timeout_id));
}
static void
storage_cancel_discover_shared_folder_callback (EvolutionStorage *storage,
const char *user,
const char *folder_name,
void *data)
{
Bonobo_Listener listener;
CORBA_Environment ev;
int timeout_id;
timeout_id = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (storage), "timeout_id"));
if (timeout_id == 0)
return;
g_source_remove (timeout_id);
g_object_set_data (G_OBJECT (storage), "timeout_id", NULL);
listener = (Bonobo_Listener) g_object_get_data (G_OBJECT (storage), "listener");
CORBA_exception_init (&ev);
Bonobo_Unknown_unref (listener, &ev);
CORBA_Object_release (listener, &ev);
CORBA_exception_free (&ev);
g_object_set_data (G_OBJECT (storage), "listener", NULL);
}
static void
storage_show_folder_properties_callback (EvolutionStorage *storage,
const char *path,
unsigned int itemNumber,
unsigned long parentWindowId,
void *data)
{
g_print ("Show properties #%d -- %s\n", itemNumber, path);
}
static void
setup_custom_storage (EvolutionShellClient *shell_client)
{
EvolutionStorage *the_storage;
EvolutionStorageResult result;
the_storage = evolution_storage_new ("TestStorage", TRUE);
g_signal_connect (the_storage, "discover_shared_folder",
G_CALLBACK (storage_discover_shared_folder_callback), shell_client);
g_signal_connect (the_storage, "cancel_discover_shared_folder",
G_CALLBACK (storage_cancel_discover_shared_folder_callback), shell_client);
/* Add some custom "Properties" items. */
evolution_storage_add_property_item (the_storage, "Sharing...",
"Change sharing properties for this folder", NULL);
evolution_storage_add_property_item (the_storage, "Permissions...",
"Change permissions for this folder", NULL);
g_signal_connect (the_storage, "show_folder_properties",
G_CALLBACK (storage_show_folder_properties_callback), NULL);
result = evolution_storage_register_on_shell (the_storage, evolution_shell_client_corba_objref (shell_client));
if (result != EVOLUTION_STORAGE_OK) {
g_warning ("Cannot register storage on the shell.");
bonobo_object_unref (BONOBO_OBJECT (the_storage));
return;
}
/* Test the sorting_priority arg here: if it was just sorting in
alphabetical order, FirstFolder would come before SecondFolder, but
we are specifying -1 sorting priority for SecondFolder and zero for
FirstFolder so the order is reversed. */
evolution_storage_new_folder (the_storage, "/FirstFolder", "FirstFolder",
"test", "file:///tmp/blah", "", "inbox", 0, TRUE, 0);
evolution_storage_new_folder (the_storage, "/SecondFolder", "SecondFolder",
"calendar", "file:///tmp/bleh", "", NULL, 0, FALSE, -1);
}
/* Callbacks. */
static void
activity_client_cancel_callback (EvolutionActivityClient *client,
void *data)
{
g_print ("User requested that the operation be cancelled.\n");
}
static void
activity_client_show_details_callback (EvolutionActivityClient *client,
void *data)
{
g_print ("User wants to see details.\n");
}
/* Timeout #3: We are done. */
static int
timeout_callback_3 (void *data)
{
EvolutionActivityClient *activity_client;
activity_client = EVOLUTION_ACTIVITY_CLIENT (data);
g_object_unref (activity_client);
g_print ("--> Done.\n");
return FALSE;
}
/* Timeout #2: Update the progress until it reaches 100%. */
static int
timeout_callback_2 (void *data)
{
EvolutionActivityClient *activity_client;
int progress;
activity_client = EVOLUTION_ACTIVITY_CLIENT (data);
progress = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (activity_client), "my_progress"));
if (progress < 0)
progress = 0;
g_print ("--> Updating %d\n", progress);
if (! evolution_activity_client_update (activity_client, "Operation Foo in progress",
(float) progress / 100.0)) {
g_warning ("Error when updating operation");
return FALSE;
}
progress ++;
g_object_set_data (G_OBJECT (activity_client), "my_progress", GINT_TO_POINTER (progress));
if (progress > 100) {
g_timeout_add (200, timeout_callback_3, activity_client);
return FALSE;
}
return TRUE;
}
/* Timeout #1: Set busy. */
static int
timeout_callback_1 (void *data)
{
EvolutionActivityClient *activity_client;
gboolean suggest_display;
GdkPixbuf *animated_icon[2];
static int count = 0;
#define NUM_ACTIVITIES 10
animated_icon[0] = gdk_pixbuf_new_from_file (EVOLUTION_IMAGES "outbox-mini.png", NULL);
animated_icon[1] = NULL;
g_assert (animated_icon[0] != NULL);
activity_client = evolution_activity_client_new (parent_shell, COMPONENT_ID,
animated_icon,
"Operation Foo started!",
TRUE,
&suggest_display);
if (activity_client == CORBA_OBJECT_NIL) {
g_warning ("Cannot create EvolutionActivityClient object");
return FALSE;
}
g_object_set_data (G_OBJECT (activity_client), "my_progress", GINT_TO_POINTER (-1));
g_signal_connect (activity_client, "cancel",
G_CALLBACK (activity_client_cancel_callback), NULL);
g_signal_connect (activity_client, "show_details",
G_CALLBACK (activity_client_show_details_callback), NULL);
g_print ("Component becoming busy -- %s\n", COMPONENT_ID);
if (suggest_display)
g_print (" --> Could display dialog box.\n");
g_timeout_add (100, timeout_callback_2, activity_client);
if (count < NUM_ACTIVITIES) {
count ++;
g_timeout_add ((rand () % 5 + 1) * 500, timeout_callback_1, NULL);
}
return FALSE;
}
static EvolutionShellComponentResult
create_view_fn (EvolutionShellComponent *shell_component,
const char *physical_uri,
const char *folder_type,
const char *view_data,
BonoboControl **control_return,
void *closure)
{
GtkWidget *vbox;
GtkWidget *label_1, *label_2, *label_3, *label_4;
GtkWidget *event_box_1, *event_box_2;
label_1 = gtk_label_new ("This is just a test component, displaying the following URI:");
label_2 = gtk_label_new (physical_uri);
if (*view_data) {
label_3 = gtk_label_new ("And the following view_data:");
label_4 = gtk_label_new (view_data);
} else
label_3 = label_4 = NULL;
event_box_1 = gtk_event_box_new ();
event_box_2 = gtk_event_box_new ();
vbox = gtk_vbox_new (FALSE, 5);
gtk_box_pack_start (GTK_BOX (vbox), event_box_1, TRUE, TRUE, 0);
gtk_box_pack_start (GTK_BOX (vbox), label_1, FALSE, TRUE, 0);
gtk_box_pack_start (GTK_BOX (vbox), label_2, FALSE, TRUE, 0);
if (label_3) {
gtk_box_pack_start (GTK_BOX (vbox), label_3, FALSE, TRUE, 0);
gtk_box_pack_start (GTK_BOX (vbox), label_4, FALSE, TRUE, 0);
}
gtk_box_pack_start (GTK_BOX (vbox), event_box_2, TRUE, TRUE, 0);
gtk_widget_show_all (vbox);
*control_return = bonobo_control_new (vbox);
g_assert (timeout_id == 0);
timeout_id = g_timeout_add (2000, timeout_callback_1, NULL);
return EVOLUTION_SHELL_COMPONENT_OK;
}
static gboolean
request_quit_fn (EvolutionShellComponent *shell_component,
void *closure)
{
GtkWidget *confirm_dialog;
GtkWidget *label;
int response;
confirm_dialog = gtk_dialog_new_with_buttons ("Quit?", NULL, GTK_DIALOG_MODAL,
GTK_STOCK_OK, GTK_RESPONSE_ACCEPT,
GTK_STOCK_CANCEL, GTK_RESPONSE_REJECT,
NULL);
label = gtk_label_new ("Please confirm that you want to quit now.");
gtk_container_add (GTK_CONTAINER (GTK_DIALOG (confirm_dialog)->vbox), label);
gtk_widget_show_all (confirm_dialog);
response = gtk_dialog_run (GTK_DIALOG (confirm_dialog));
gtk_widget_destroy (confirm_dialog);
if (response == GTK_RESPONSE_ACCEPT)
return TRUE; /* OK */
else
return FALSE; /* Cancel */
}
/* Callbacks. */
static void
owner_set_callback (EvolutionShellComponent *shell_component,
EvolutionShellClient *shell_client,
const char *evolution_homedir)
{
g_assert (parent_shell == NULL);
g_print ("We have an owner -- home directory is `%s'\n", evolution_homedir);
parent_shell = shell_client;
if (evolution_shell_client_get_activity_interface (parent_shell) == CORBA_OBJECT_NIL)
g_warning ("Shell doesn't have a ::Activity interface -- weird!");
spit_out_shortcuts (shell_client);
setup_custom_storage (shell_client);
}
static int
owner_unset_idle_callback (void *data)
{
gtk_main_quit ();
return FALSE;
}
static void
owner_unset_callback (EvolutionShellComponent *shell_component,
void *data)
{
g_idle_add_full (G_PRIORITY_LOW, owner_unset_idle_callback, NULL, NULL);
}
static void
user_create_new_item_callback (EvolutionShellComponent *shell_component,
const char *id,
const char *parent_folder_physical_uri,
const char *parent_folder_type)
{
g_print ("\n*** Should create -- %s\n", id);
g_print ("\n\tType %s, URI %s\n", parent_folder_type, parent_folder_physical_uri);
if (strcmp (id, "FolderSelector") == 0)
create_new_folder_selector (shell_component);
}
static void
register_component (void)
{
EvolutionShellComponent *shell_component;
int result;
shell_component = evolution_shell_component_new (folder_types,
NULL,
create_view_fn,
NULL, NULL, NULL, NULL, NULL, NULL,
request_quit_fn,
NULL);
g_signal_connect (shell_component, "owner_set",
G_CALLBACK (owner_set_callback), NULL);
g_signal_connect (shell_component, "owner_unset",
G_CALLBACK (owner_unset_callback), NULL);
evolution_shell_component_add_user_creatable_item (shell_component, "Stuff",
"New Stuff", "New _Stuff",
"Create some new stuff",
NULL,
'\0', NULL);
evolution_shell_component_add_user_creatable_item (shell_component, "MoreStuff",
"New More Stuff", "New _More Stuff",
"Create more stuff",
NULL,
'n', NULL);
evolution_shell_component_add_user_creatable_item (shell_component, "FolderSelector",
"Folder Selector", "New Folder _Selector",
"Show a folder selector",
NULL,
's', NULL);
g_signal_connect (shell_component, "user_create_new_item",
G_CALLBACK (user_create_new_item_callback), NULL);
result = bonobo_activation_active_server_register (COMPONENT_ID,
bonobo_object_corba_objref (BONOBO_OBJECT (shell_component)));
if (result != Bonobo_ACTIVATION_REG_SUCCESS)
g_error ("Cannot register active server into OAF");
}
int
main (int argc, char **argv)
{
bindtextdomain (GETTEXT_PACKAGE, EVOLUTION_LOCALEDIR);
textdomain (GETTEXT_PACKAGE);
gnome_program_init (PACKAGE, VERSION, LIBGNOMEUI_MODULE, argc, argv,
GNOME_PROGRAM_STANDARD_PROPERTIES,
GNOME_PARAM_HUMAN_READABLE_NAME, _("Evolution Test Component"),
NULL);
register_configuration_control_factory ();
register_component ();
g_print ("Test Component up and running.\n");
bonobo_main ();
return 0;
}