diff options
-rw-r--r-- | plugins/calendar-weather/ChangeLog | 3 | ||||
-rw-r--r-- | plugins/calendar-weather/Makefile.am | 29 | ||||
-rw-r--r-- | plugins/calendar-weather/calendar-weather.c | 764 | ||||
-rw-r--r-- | plugins/calendar-weather/category_weather_cloudy_16.png | bin | 0 -> 597 bytes | |||
-rw-r--r-- | plugins/calendar-weather/category_weather_fog_16.png | bin | 0 -> 217 bytes | |||
-rw-r--r-- | plugins/calendar-weather/category_weather_partly_cloudy_16.png | bin | 0 -> 760 bytes | |||
-rw-r--r-- | plugins/calendar-weather/category_weather_rain_16.png | bin | 0 -> 647 bytes | |||
-rw-r--r-- | plugins/calendar-weather/category_weather_snow_16.png | bin | 0 -> 624 bytes | |||
-rw-r--r-- | plugins/calendar-weather/category_weather_sun_16.png | bin | 0 -> 420 bytes | |||
-rw-r--r-- | plugins/calendar-weather/category_weather_tstorm_16.png | bin | 0 -> 728 bytes | |||
-rw-r--r-- | plugins/calendar-weather/org-gnome-calendar-weather.eplug.in | 37 |
11 files changed, 833 insertions, 0 deletions
diff --git a/plugins/calendar-weather/ChangeLog b/plugins/calendar-weather/ChangeLog new file mode 100644 index 0000000000..5615b5425e --- /dev/null +++ b/plugins/calendar-weather/ChangeLog @@ -0,0 +1,3 @@ +2004-01-06 David Trowbridge <trowbrds@cs.colorado.edu> + + * Initial import of weather properties plugin diff --git a/plugins/calendar-weather/Makefile.am b/plugins/calendar-weather/Makefile.am new file mode 100644 index 0000000000..ef17a5314b --- /dev/null +++ b/plugins/calendar-weather/Makefile.am @@ -0,0 +1,29 @@ +INCLUDES = \ + -I$(top_srcdir)/shell \ + $(EVOLUTION_CFLAGS) \ + $(EVOLUTION_CALENDAR_CFLAGS) \ + $(SOURCE_SEL_CFLAGS) \ + -DWEATHER_DATADIR=\""$(weatherdatadir)"\" + +@EVO_PLUGIN_RULE@ + +# These will actually install into the e-d-s share directory, but +# if the plugin is disabled, we aren't going to be needing them. +weatherdata_DATA = \ + category_weather_cloudy_16.png \ + category_weather_fog_16.png \ + category_weather_partly_cloudy_16.png \ + category_weather_rain_16.png \ + category_weather_snow_16.png \ + category_weather_sun_16.png \ + category_weather_tstorm_16.png + +plugin_DATA = org-gnome-calendar-weather.eplug +plugin_LTLIBRARIES = liborg-gnome-calendar-weather.la + +liborg_gnome_calendar_weather_la_SOURCES = calendar-weather.c +liborg_gnome_calendar_weather_la_LDFLAGS = -module -avoid-version + +EXTRA_DIST = \ + org-gnome-calendar-weather.eplug.in \ + $(weatherdata_DATA) diff --git a/plugins/calendar-weather/calendar-weather.c b/plugins/calendar-weather/calendar-weather.c new file mode 100644 index 0000000000..d7c68b8141 --- /dev/null +++ b/plugins/calendar-weather/calendar-weather.c @@ -0,0 +1,764 @@ +/* + * Authors: David Trowbridge <trowbrds@cs.colorado.edu> + * + * Copyright (C) 2005 Novell, Inc. (www.novell.com) + * + * 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. + * + */ + +#include <gtk/gtk.h> +#include <e-util/e-config.h> +#include <calendar/gui/e-cal-config.h> +#include <calendar/gui/e-cal-event.h> +#include <calendar/gui/calendar-component.h> +#include <libedataserver/e-source.h> +#include <libedataserver/e-url.h> +#include <libedataserver/e-categories.h> +#include <libgnome/gnome-i18n.h> +#include <libxml/parser.h> +#include <libxml/tree.h> +#include <string.h> + +GtkWidget *e_calendar_weather_location (EPlugin *epl, EConfigHookItemFactoryData *data); +GtkWidget *e_calendar_weather_refresh (EPlugin *epl, EConfigHookItemFactoryData *data); +GtkWidget *e_calendar_weather_temperature (EPlugin *epl, EConfigHookItemFactoryData *data); +GtkWidget *e_calendar_weather_snowfall (EPlugin *epl, EConfigHookItemFactoryData *data); +gboolean e_calendar_weather_check (EPlugin *epl, EConfigHookPageCheckData *data); +void e_calendar_weather_migrate (EPlugin *epl, ECalEventTargetComponent *data); +int e_plugin_lib_enable (EPluginLib *epl, int enable); + +static GtkTreeStore *store = NULL; + +#define WEATHER_BASE_URI "weather://" + +int +e_plugin_lib_enable (EPluginLib *epl, int enable) +{ + GList *l; + gboolean found = FALSE; + + /* Add the categories icons if we don't have them. */ + for (l = e_categories_get_list (); l; l = g_list_next (l)) { + if (!strcmp (l->data, _("Weather: Cloudy"))) { + found = TRUE; + break; + } + } + if (!found) { + e_categories_add (_("Weather: Cloudy"), NULL, WEATHER_DATADIR "/category_weather_cloudy_16.png"); + e_categories_add (_("Weather: Fog"), NULL, WEATHER_DATADIR "/category_weather_fog_16.png"); + e_categories_add (_("Weather: Partly Cloudy"), NULL, WEATHER_DATADIR "/category_weather_partly_cloudy_16.png"); + e_categories_add (_("Weather: Rain"), NULL, WEATHER_DATADIR "/category_weather_rain_16.png"); + e_categories_add (_("Weather: Snow"), NULL, WEATHER_DATADIR "/category_weather_snow_16.png"); + e_categories_add (_("Weather: Sunny"), NULL, WEATHER_DATADIR "/category_weather_sun_16.png"); + e_categories_add (_("Weather: Thunderstorms"), NULL, WEATHER_DATADIR "/category_weather_tstorm_16.png"); + } + + return 0; +} + +void +e_calendar_weather_migrate (EPlugin *epl, ECalEventTargetComponent *data) +{ + /* Perform a migration step here. This allows us to keep the weather calendar completely + * separate from evolution. If the plugin isn't built, the weather source group won't + * show up in the user's evolution. If it is, this will create it if it doesn't exist */ + CalendarComponent *component; + ESourceList *source_list; + ESourceGroup *group; + GSList *groups; + ESourceGroup *weather = NULL; + + component = data->component; + source_list = calendar_component_peek_source_list (component); + + groups = e_source_list_peek_groups (source_list); + if (groups) { + /* groups are already there, we need to search */ + GSList *g; + + for (g = groups; g; g = g_slist_next (g)) { + group = E_SOURCE_GROUP (g->data); + if (!weather && !strcmp (WEATHER_BASE_URI, e_source_group_peek_base_uri (group))) + weather = g_object_ref (group); + } + } + + if (!weather) { + group = e_source_group_new (_("Weather"), WEATHER_BASE_URI); + e_source_list_add_group (source_list, group, -1); + + weather = group; + } + + if (weather) + g_object_unref (weather); + + e_source_list_sync (source_list, NULL); + return 0; +} + +static void +parse_subtree (GtkTreeIter *parent, xmlNode *node) +{ + GtkTreeIter iter; + xmlNode *child; + + if (node->type == XML_ELEMENT_NODE) { + gtk_tree_store_append (store, &iter, parent); + if (strcmp (node->name, "location") == 0) { + xmlAttr *attr; + + child = node->children; + g_assert (child->type == XML_TEXT_NODE); + gtk_tree_store_set (store, &iter, 0, child->content, -1); + + for (attr = node->properties; attr; attr = attr->next) { + if (strcmp (attr->name, "code") == 0) + gtk_tree_store_set (store, &iter, 1, attr->children->content, -1); + else if (strcmp (attr->name, "url") == 0) + gtk_tree_store_set (store, &iter, 2, attr->children->content, -1); + else if (strcmp (attr->name, "type") == 0) + gtk_tree_store_set (store, &iter, 3, attr->children->content, -1); + } + } else { + xmlAttr *attr; + + for (child = node->children; child; child = child->next) + parse_subtree (&iter, child); + + for (attr = node->properties; attr; attr = attr->next) + if (strcmp (attr->name, "name") == 0) + gtk_tree_store_set (store, &iter, 0, attr->children->content, -1); + } + } +} + +static void +load_locations () +{ + xmlDoc *doc; + xmlNode *root, *child; + + LIBXML_TEST_VERSION + + doc = xmlParseFile (WEATHER_DATADIR "/Locations.xml"); + if (doc == NULL) { + g_warning ("failed to read locations file"); + return; + } + + if (store == NULL) + store = gtk_tree_store_new (4, + G_TYPE_STRING, /* name */ + G_TYPE_STRING, /* code */ + G_TYPE_STRING, /* URL */ + G_TYPE_STRING); /* type */ + + root = xmlDocGetRootElement (doc); + for (child = root->children; child; child = child->next) + parse_subtree (NULL, child); + xmlFreeDoc (doc); +} + +static void +selection_changed (GtkTreeSelection *selection, GtkDialog *dialog) +{ + GtkTreeModel *model; + GtkTreeIter iter; + + if (gtk_tree_selection_get_selected (selection, &model, &iter)) { + gchar *code = NULL; + gtk_tree_model_get (model, &iter, 1, &code, -1); + if (code != NULL) { + gtk_dialog_set_response_sensitive (GTK_DIALOG (dialog), GTK_RESPONSE_OK, TRUE); + } else { + GtkTreeView *view = gtk_tree_selection_get_tree_view (selection); + GtkTreePath *path; + path = gtk_tree_model_get_path (model, &iter); + gtk_tree_view_expand_row (view, path, FALSE); + gtk_tree_path_free (path); + gtk_dialog_set_response_sensitive (GTK_DIALOG (dialog), GTK_RESPONSE_OK, FALSE); + } + } else { + gtk_dialog_set_response_sensitive (GTK_DIALOG (dialog), GTK_RESPONSE_OK, FALSE); + } +} + +static struct +{ + gchar **ids; + GtkTreeIter *result; +} find_data; + +static gboolean +find_location_func (GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *node, gpointer data) +{ + gchar *type, *code, *name; + gtk_tree_model_get (model, node, 0, &name, 1, &code, 3, &type, -1); + if (name == NULL || code == NULL || type == NULL) + return FALSE; + if ((!strcmp (type, find_data.ids[0])) && + (!strcmp (code, find_data.ids[1])) && + (!strcmp (name, find_data.ids[2]))) { + find_data.result = gtk_tree_iter_copy (node); + return TRUE; + } + return FALSE; +} + +static GtkTreeIter * +find_location (gchar *relative_url) +{ + /* type/code/name */ + find_data.ids = g_strsplit (relative_url, "/", -1); + find_data.result = NULL; + gtk_tree_model_foreach (GTK_TREE_MODEL (store), (GtkTreeModelForeachFunc) find_location_func, NULL); + + g_strfreev (find_data.ids); + return find_data.result; +} + +static gboolean +treeview_clicked (GtkTreeView *treeview, GdkEventButton *event, GtkDialog *dialog) +{ + if (event->type == GDK_2BUTTON_PRESS) { + GtkTreeSelection *selection = gtk_tree_view_get_selection (treeview); + GtkTreeModel *model; + GtkTreeIter iter; + + if (gtk_tree_selection_get_selected (selection, &model, &iter)) { + gchar *code = NULL; + gtk_tree_model_get (model, &iter, 1, &code, -1); + if (code != NULL) { + gtk_dialog_response (dialog, GTK_RESPONSE_OK); + return TRUE; + } + } + } + return FALSE; +} + +static GtkDialog * +create_source_selector (ESource *source) +{ + GtkWidget *dialog, *treeview, *scrolledwindow; + GtkCellRenderer *text; + GtkTreeSelection *selection; + gchar *uri_text; + EUri *uri; + + /* FIXME - should show an error here if it fails*/ + if (store == NULL) + return NULL; + + dialog = gtk_dialog_new_with_buttons ( + _("Select a location"), + NULL, GTK_DIALOG_MODAL, + GTK_STOCK_OK, GTK_RESPONSE_OK, + GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, + NULL); + gtk_container_set_border_width (GTK_CONTAINER (dialog), 6); + + scrolledwindow = gtk_scrolled_window_new (NULL, NULL); + gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolledwindow), GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC); + gtk_widget_show (scrolledwindow); + treeview = gtk_tree_view_new_with_model (GTK_TREE_MODEL (store)); + gtk_tree_view_set_headers_visible (GTK_TREE_VIEW (treeview), FALSE); + gtk_widget_show (treeview); + gtk_scrolled_window_add_with_viewport (GTK_SCROLLED_WINDOW (scrolledwindow), treeview); + gtk_widget_add_events (treeview, GDK_BUTTON_PRESS); + g_signal_connect (G_OBJECT (treeview), "button-press-event", G_CALLBACK (treeview_clicked), dialog); + selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (treeview)); + gtk_tree_selection_set_mode (selection, GTK_SELECTION_BROWSE); + + uri_text = e_source_get_uri (source); + uri = e_uri_new (uri_text); + if (uri->path && strlen (uri->path)) { + GtkTreeIter *iter = find_location (uri_text + 10); + GtkTreePath *path = gtk_tree_model_get_path (GTK_TREE_MODEL (store), iter); + gtk_tree_view_expand_to_path (GTK_TREE_VIEW (treeview), path); + gtk_tree_selection_select_path (selection, path); + gtk_tree_path_free (path); + } + g_free (uri_text); + e_uri_free (uri); + + g_signal_connect (G_OBJECT (selection), "changed", G_CALLBACK (selection_changed), dialog); + g_object_set_data (G_OBJECT (dialog), "treeview", treeview); + + text = gtk_cell_renderer_text_new (); + gtk_tree_view_insert_column_with_attributes (GTK_TREE_VIEW (treeview), -1, "location", text, "text", 0, NULL); + + gtk_container_add (GTK_CONTAINER (GTK_DIALOG (dialog)->vbox), scrolledwindow); + gtk_container_set_border_width (GTK_CONTAINER (scrolledwindow), 6); + gtk_box_set_spacing (GTK_BOX (GTK_DIALOG (dialog)->vbox), 6); + + gtk_dialog_set_response_sensitive (GTK_DIALOG (dialog), GTK_RESPONSE_OK, FALSE); + gtk_window_set_default_size (GTK_WINDOW (dialog), 420, 340); + + return GTK_DIALOG (dialog); +} + +static gchar * +build_location_path (GtkTreeIter *iter) +{ + GtkTreeIter parent; + gchar *path, *temp1, *temp2; + + gtk_tree_model_get (GTK_TREE_MODEL (store), iter, 0, &temp1, -1); + path = g_strdup (temp1); + + while (gtk_tree_model_iter_parent (GTK_TREE_MODEL (store), &parent, iter)) { + gtk_tree_model_get (GTK_TREE_MODEL (store), &parent, 0, &temp1, -1); + temp2 = g_strdup_printf ("%s : %s", temp1, path); + g_free (path); + path = temp2; + iter = gtk_tree_iter_copy (&parent); + } + return path; +} + +static void +location_clicked (GtkButton *button, ESource *source) +{ + GtkDialog *dialog = create_source_selector (source); + gint response; + + if (dialog == NULL) + return; + + response = gtk_dialog_run (dialog); + + if (response == GTK_RESPONSE_OK) { + GtkTreeView *view = GTK_TREE_VIEW (g_object_get_data (G_OBJECT (dialog), "treeview")); + GtkTreeSelection *selection = gtk_tree_view_get_selection (view); + GtkTreeModel *model; + GtkTreeIter iter; + GtkWidget *label; + gchar *type, *code, *name; + gchar *path, *uri; + + gtk_tree_selection_get_selected (selection, &model, &iter); + gtk_tree_model_get (model, &iter, 0, &name, 1, &code, 3, &type, -1); + path = build_location_path (&iter); + + label = gtk_bin_get_child (GTK_BIN (button)); + gtk_label_set_text (GTK_LABEL (label), path); + + uri = g_strdup_printf ("%s/%s/%s", type, code, name); + /* FIXME - url_encode (&uri); */ + e_source_set_relative_uri (source, uri); + g_free (uri); + } else { + GtkWidget *label; + const gchar *text; + + label = GTK_WIDGET (gtk_bin_get_child (GTK_BIN (button))); + text = gtk_label_get_text (GTK_LABEL (label)); + if (strcmp (text, _("None")) == 0) + e_source_set_relative_uri (source, ""); + } + + gtk_widget_destroy (GTK_WIDGET (dialog)); +} + +GtkWidget * +e_calendar_weather_location (EPlugin *epl, EConfigHookItemFactoryData *data) +{ + static GtkWidget *label; + GtkWidget *button, *parent, *text; + int row; + ECalConfigTargetSource *t = (ECalConfigTargetSource *) data->target; + ESource *source = t->source; + EUri *uri; + char *uri_text; + static GtkWidget *hidden; + + if (store == NULL) + load_locations (); + + if (!hidden) + hidden = gtk_label_new (""); + + if (data->old) + gtk_widget_destroy (label); + + uri_text = e_source_get_uri (t->source); + uri = e_uri_new (uri_text); + if (strcmp (uri->protocol, "weather")) { + e_uri_free (uri); + return hidden; + } + + parent = data->parent; + + row = ((GtkTable*)parent)->nrows; + + label = gtk_label_new_with_mnemonic (_("_Location:")); + gtk_widget_show (label); + gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); + gtk_table_attach (GTK_TABLE (parent), label, 0, 1, row, row+1, GTK_FILL, 0, 0, 0); + + button = gtk_button_new (); + g_signal_connect (G_OBJECT (button), "clicked", G_CALLBACK (location_clicked), source); + gtk_widget_show (button); + + if (uri->path && strlen (uri->path)) { + GtkTreeIter *iter = find_location (uri_text + 10); + gchar *location = build_location_path (iter); + text = gtk_label_new (location); + g_free (location); + } else + text = gtk_label_new (_("None")); + gtk_widget_show (text); +#if (GTK_CHECK_VERSION(2, 6, 0)) + gtk_label_set_ellipsize (GTK_LABEL (text), PANGO_ELLIPSIZE_START); +#endif + gtk_container_add (GTK_CONTAINER (button), text); + e_uri_free (uri); + g_free (uri_text); + + gtk_table_attach (GTK_TABLE (parent), button, 1, 2, row, row+1, GTK_EXPAND | GTK_FILL, 0, 0, 0); + + return button; +} + +static void +set_refresh_time (ESource *source, GtkWidget *spin, GtkWidget *option) +{ + int time; + int item_num = 0; + const char *refresh_str = e_source_get_property (source, "refresh"); + time = refresh_str ? atoi (refresh_str) : 30; + + if (time && !(time % 10080)) { + /* weeks */ + item_num = 3; + time /= 10080; + } else if (time && !(time % 1440)) { + /* days */ + item_num = 2; + time /= 1440; + } else if (time && !(time % 60)) { + /* hours */ + item_num = 1; + time /= 60; + } + gtk_option_menu_set_history (GTK_OPTION_MENU (option), item_num); + gtk_spin_button_set_value (GTK_SPIN_BUTTON (spin), time); +} + +static char * +get_refresh_minutes (GtkWidget *spin, GtkWidget *option) +{ + int setting = gtk_spin_button_get_value_as_int (GTK_SPIN_BUTTON (spin)); + switch (gtk_option_menu_get_history (GTK_OPTION_MENU (option))) { + case 0: + /* minutes */ + break; + case 1: + /* hours */ + setting *= 60; + break; + case 2: + /* days */ + setting *= 1440; + break; + case 3: + /* weeks - is this *really* necessary? */ + setting *= 10080; + break; + default: + g_warning ("Time unit out of range"); + break; + } + return g_strdup_printf ("%d", setting); +} + +static void +spin_changed (GtkSpinButton *spin, ECalConfigTargetSource *t) +{ + char *refresh_str; + GtkWidget *option; + + option = g_object_get_data (G_OBJECT (spin), "option"); + + refresh_str = get_refresh_minutes ((GtkWidget *) spin, option); + e_source_set_property (t->source, "refresh", refresh_str); + g_free (refresh_str); +} + +static void +option_changed (GtkOptionMenu *option, ECalConfigTargetSource *t) +{ + char *refresh_str; + GtkWidget *spin; + + spin = g_object_get_data (G_OBJECT (option), "spin"); + + refresh_str = get_refresh_minutes (spin, (GtkWidget *) option); + e_source_set_property (t->source, "refresh", refresh_str); + g_free (refresh_str); +} + +GtkWidget * +e_calendar_weather_refresh (EPlugin *epl, EConfigHookItemFactoryData *data) +{ + static GtkWidget *label; + GtkWidget *option, *spin, *menu, *hbox, *parent; + GtkWidget *times[4]; + int row, i; + ECalConfigTargetSource *t = (ECalConfigTargetSource *) data->target; + ESource *source = t->source; + EUri *uri; + char *uri_text; + static GtkWidget *hidden = NULL; + + if (!hidden) + hidden = gtk_label_new (""); + + if (data->old) + gtk_widget_destroy (label); + + uri_text = e_source_get_uri (t->source); + uri = e_uri_new (uri_text); + g_free (uri_text); + if (strcmp (uri->protocol, "weather")) { + e_uri_free (uri); + return hidden; + } + e_uri_free (uri); + + parent = data->parent; + + row = ((GtkTable*)parent)->nrows; + + label = gtk_label_new_with_mnemonic (_("_Refresh:")); + gtk_widget_show (label); + gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); + gtk_table_attach (GTK_TABLE (parent), label, 0, 1, row, row+1, GTK_FILL, 0, 0, 0); + + hbox = gtk_hbox_new (FALSE, 6); + gtk_widget_show (hbox); + + spin = gtk_spin_button_new_with_range (0, 100, 1); + gtk_label_set_mnemonic_widget (GTK_LABEL (label), spin); + gtk_widget_show (spin); + gtk_box_pack_start (GTK_BOX (hbox), spin, FALSE, TRUE, 0); + + option = gtk_option_menu_new (); + gtk_widget_show (option); + times[0] = gtk_menu_item_new_with_label (_("minutes")); + times[1] = gtk_menu_item_new_with_label (_("hours")); + times[2] = gtk_menu_item_new_with_label (_("days")); + times[3] = gtk_menu_item_new_with_label (_("weeks")); + menu = gtk_menu_new (); + gtk_widget_show (menu); + for (i = 0; i < 4; i++) { + gtk_widget_show (times[i]); + gtk_menu_shell_append (GTK_MENU_SHELL (menu), times[i]); + } + gtk_option_menu_set_menu (GTK_OPTION_MENU (option), menu); + set_refresh_time (source, spin, option); + gtk_box_pack_start (GTK_BOX (hbox), option, FALSE, TRUE, 0); + + g_object_set_data (G_OBJECT (option), "spin", spin); + g_signal_connect (G_OBJECT (option), "changed", G_CALLBACK (option_changed), t); + g_object_set_data (G_OBJECT (spin), "option", option); + g_signal_connect (G_OBJECT (spin), "value-changed", G_CALLBACK (spin_changed), t); + + gtk_table_attach (GTK_TABLE (parent), hbox, 1, 2, row, row+1, GTK_EXPAND | GTK_FILL, 0, 0, 0); + + return hbox; +} + +static void +set_temperature_units (ESource *source, GtkWidget *option) +{ + const char *format = e_source_get_property (source, "temperature"); + if (format == NULL) + gtk_option_menu_set_history (GTK_OPTION_MENU (option), 0); + else if (strcmp (format, "fahrenheit") == 0) + gtk_option_menu_set_history (GTK_OPTION_MENU (option), 1); + else + gtk_option_menu_set_history (GTK_OPTION_MENU (option), 0); +} + +static void +temperature_units_changed (GtkOptionMenu *option, ECalConfigTargetSource *t) +{ + int choice = gtk_option_menu_get_history (GTK_OPTION_MENU (option)); + if (choice == 0) + e_source_set_property (t->source, "temperature", "celcius"); + else + e_source_set_property (t->source, "temperature", "fahrenheit"); +} + +GtkWidget * +e_calendar_weather_temperature (EPlugin *epl, EConfigHookItemFactoryData *data) +{ + static GtkWidget *label; + GtkWidget *option, *menu, *parent; + GtkWidget *formats[2]; + int row, i; + ECalConfigTargetSource *t = (ECalConfigTargetSource *) data->target; + ESource *source = t->source; + EUri *uri; + char *uri_text; + static GtkWidget *hidden = NULL; + + if (!hidden) + hidden = gtk_label_new (""); + + if (data->old) + gtk_widget_destroy (label); + + uri_text = e_source_get_uri (t->source); + uri = e_uri_new (uri_text); + g_free (uri_text); + if (strcmp (uri->protocol, "weather")) { + e_uri_free (uri); + return hidden; + } + e_uri_free (uri); + + parent = data->parent; + + row = ((GtkTable*)parent)->nrows; + + label = gtk_label_new_with_mnemonic (_("_Temperature Units:")); + gtk_widget_show (label); + gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); + gtk_table_attach (GTK_TABLE (parent), label, 0, 1, row, row+1, GTK_FILL, 0, 0, 0); + + option = gtk_option_menu_new (); + gtk_widget_show (option); + formats[0] = gtk_menu_item_new_with_label (_("Celcius")); + formats[1] = gtk_menu_item_new_with_label (_("Fahrenheit")); + menu = gtk_menu_new (); + gtk_widget_show (menu); + for (i = 0; i < 2; i++) { + gtk_widget_show (formats[i]); + gtk_menu_shell_append (GTK_MENU_SHELL (menu), formats[i]); + } + gtk_option_menu_set_menu (GTK_OPTION_MENU (option), menu); + set_temperature_units (source, option); + gtk_label_set_mnemonic_widget (GTK_LABEL (label), option); + g_signal_connect (G_OBJECT (option), "changed", G_CALLBACK (temperature_units_changed), t); + gtk_table_attach (GTK_TABLE (parent), option, 1, 2, row, row+1, GTK_FILL, 0, 0, 0); + + return option; +} + +static void +set_snowfall_units (ESource *source, GtkWidget *option) +{ + const char *format = e_source_get_property (source, "snowfall"); + if (format == NULL) + gtk_option_menu_set_history (GTK_OPTION_MENU (option), 0); + else if (strcmp (format, "inches") == 0) + gtk_option_menu_set_history (GTK_OPTION_MENU (option), 1); + else + gtk_option_menu_set_history (GTK_OPTION_MENU (option), 0); +} + +static void +snowfall_units_changed (GtkOptionMenu *option, ECalConfigTargetSource *t) +{ + int choice = gtk_option_menu_get_history (GTK_OPTION_MENU (option)); + if (choice == 0) + e_source_set_property (t->source, "snowfall", "centimeters"); + else + e_source_set_property (t->source, "snowfall", "inches"); +} + +GtkWidget * +e_calendar_weather_snowfall (EPlugin *epl, EConfigHookItemFactoryData *data) +{ + static GtkWidget *label; + GtkWidget *option, *menu, *parent; + GtkWidget *formats[2]; + int row, i; + ECalConfigTargetSource *t = (ECalConfigTargetSource *) data->target; + ESource *source = t->source; + EUri *uri; + char *uri_text; + static GtkWidget *hidden = NULL; + + if (!hidden) + hidden = gtk_label_new (""); + + if (data->old) + gtk_widget_destroy (label); + + uri_text = e_source_get_uri (t->source); + uri = e_uri_new (uri_text); + g_free (uri_text); + if (strcmp (uri->protocol, "weather")) { + e_uri_free (uri); + return hidden; + } + e_uri_free (uri); + + parent = data->parent; + + row = ((GtkTable*)parent)->nrows; + + label = gtk_label_new_with_mnemonic (_("_Snowfall Units:")); + gtk_widget_show (label); + gtk_misc_set_alignment (GTK_MISC (label), 0.0, 0.5); + gtk_table_attach (GTK_TABLE (parent), label, 0, 1, row, row+1, GTK_FILL, 0, 0, 0); + + option = gtk_option_menu_new (); + gtk_widget_show (option); + formats[0] = gtk_menu_item_new_with_label (_("Centimeters")); + formats[1] = gtk_menu_item_new_with_label (_("Inches")); + menu = gtk_menu_new (); + gtk_widget_show (menu); + for (i = 0; i < 2; i++) { + gtk_widget_show (formats[i]); + gtk_menu_shell_append (GTK_MENU_SHELL (menu), formats[i]); + } + gtk_option_menu_set_menu (GTK_OPTION_MENU (option), menu); + set_snowfall_units (source, option); + gtk_label_set_mnemonic_widget (GTK_LABEL (label), option); + g_signal_connect (G_OBJECT (option), "changed", G_CALLBACK (snowfall_units_changed), t); + gtk_table_attach (GTK_TABLE (parent), option, 1, 2, row, row+1, GTK_FILL, 0, 0, 0); + + return option; +} + +gboolean +e_calendar_weather_check (EPlugin *epl, EConfigHookPageCheckData *data) +{ + /* FIXME - check pageid */ + ECalConfigTargetSource *t = (ECalConfigTargetSource *) data->target; + EUri *uri; + gboolean ok = FALSE; + ESourceGroup *group = e_source_peek_group (t->source); + + /* always return TRUE if this isn't a weather source */ + if (strncmp (e_source_group_peek_base_uri (group), "weather", 7)) + return TRUE; + + uri = e_uri_new (e_source_get_uri (t->source)); + /* make sure that the protocol is weather:// and that the path isn't empty */ + ok = (uri->path && strlen (uri->path)); + e_uri_free (uri); + + return ok; +} diff --git a/plugins/calendar-weather/category_weather_cloudy_16.png b/plugins/calendar-weather/category_weather_cloudy_16.png Binary files differnew file mode 100644 index 0000000000..ddb3ba7c59 --- /dev/null +++ b/plugins/calendar-weather/category_weather_cloudy_16.png diff --git a/plugins/calendar-weather/category_weather_fog_16.png b/plugins/calendar-weather/category_weather_fog_16.png Binary files differnew file mode 100644 index 0000000000..23e4e2f1d4 --- /dev/null +++ b/plugins/calendar-weather/category_weather_fog_16.png diff --git a/plugins/calendar-weather/category_weather_partly_cloudy_16.png b/plugins/calendar-weather/category_weather_partly_cloudy_16.png Binary files differnew file mode 100644 index 0000000000..472feaa654 --- /dev/null +++ b/plugins/calendar-weather/category_weather_partly_cloudy_16.png diff --git a/plugins/calendar-weather/category_weather_rain_16.png b/plugins/calendar-weather/category_weather_rain_16.png Binary files differnew file mode 100644 index 0000000000..e00d5e1c82 --- /dev/null +++ b/plugins/calendar-weather/category_weather_rain_16.png diff --git a/plugins/calendar-weather/category_weather_snow_16.png b/plugins/calendar-weather/category_weather_snow_16.png Binary files differnew file mode 100644 index 0000000000..5e95985f5f --- /dev/null +++ b/plugins/calendar-weather/category_weather_snow_16.png diff --git a/plugins/calendar-weather/category_weather_sun_16.png b/plugins/calendar-weather/category_weather_sun_16.png Binary files differnew file mode 100644 index 0000000000..780c61c23c --- /dev/null +++ b/plugins/calendar-weather/category_weather_sun_16.png diff --git a/plugins/calendar-weather/category_weather_tstorm_16.png b/plugins/calendar-weather/category_weather_tstorm_16.png Binary files differnew file mode 100644 index 0000000000..b2af092b53 --- /dev/null +++ b/plugins/calendar-weather/category_weather_tstorm_16.png diff --git a/plugins/calendar-weather/org-gnome-calendar-weather.eplug.in b/plugins/calendar-weather/org-gnome-calendar-weather.eplug.in new file mode 100644 index 0000000000..9a9357d0b9 --- /dev/null +++ b/plugins/calendar-weather/org-gnome-calendar-weather.eplug.in @@ -0,0 +1,37 @@ +<?xml version="1.0"?> +<e-plugin-list> + <e-plugin + type="shlib" + location="@PLUGINDIR@/liborg-gnome-calendar-weather.so" + id="org.gnome.evolution.calendar.weather" + name="Weather Calendars"> + <description>Provides core functionality for weather calendars</description> + <author name="David Trowbridge" email="trowbrds@cs.colorado.edu"/> + <hook class="org.gnome.evolution.calendar.config:1.0"> + <group + target="source" + id="org.gnome.evolution.calendar.calendarProperties" + check="e_calendar_weather_check"> + <item + type="item_table" + path="00.general/00.source/40.location" + factory="e_calendar_weather_location"/> + <item + type="item_table" + path="00.general/00.source/50.tempFormat" + factory="e_calendar_weather_temperature"/> + <item + type="item_table" + path="00.general/00.source/60.snowfallFormat" + factory="e_calendar_weather_snowfall"/> + <item + type="item_table" + path="00.general/00.source/70.refresh" + factory="e_calendar_weather_refresh"/> + </group> + </hook> + <hook class="org.gnome.evolution.calendar.events:1.0"> + <event target="component" id="component.migration" handle="e_calendar_weather_migrate"/> + </hook> + </e-plugin> +</e-plugin-list> |