aboutsummaryrefslogtreecommitdiffstats
path: root/calendar/gui/cal-search-bar.c
diff options
context:
space:
mode:
Diffstat (limited to 'calendar/gui/cal-search-bar.c')
-rw-r--r--calendar/gui/cal-search-bar.c279
1 files changed, 140 insertions, 139 deletions
diff --git a/calendar/gui/cal-search-bar.c b/calendar/gui/cal-search-bar.c
index f02528a8eb..4c0937097f 100644
--- a/calendar/gui/cal-search-bar.c
+++ b/calendar/gui/cal-search-bar.c
@@ -47,7 +47,8 @@ enum {
SEARCH_ANY_FIELD_CONTAINS,
SEARCH_SUMMARY_CONTAINS,
SEARCH_DESCRIPTION_CONTAINS,
- SEARCH_COMMENT_CONTAINS
+ SEARCH_COMMENT_CONTAINS,
+ SEARCH_CATEGORY_IS,
};
static ESearchBarItem search_option_items[] = {
@@ -55,13 +56,19 @@ static ESearchBarItem search_option_items[] = {
{ N_("Summary contains"), SEARCH_SUMMARY_CONTAINS, NULL },
{ N_("Description contains"), SEARCH_DESCRIPTION_CONTAINS, NULL },
{ N_("Comment contains"), SEARCH_COMMENT_CONTAINS, NULL },
+ { N_("Category is"), SEARCH_CATEGORY_IS, NULL },
{ NULL, -1, NULL }
};
+/* IDs for the categories suboptions */
+#define CATEGORIES_ALL 0
+#define CATEGORIES_UNMATCHED 1
+#define CATEGORIES_OFFSET 2
+
/* Private part of the CalSearchBar structure */
struct CalSearchBarPrivate {
- /* Option menu for the categories drop-down */
- GtkOptionMenu *categories_omenu;
+ /* Array of categories */
+ GPtrArray *categories;
};
@@ -167,7 +174,22 @@ cal_search_bar_init (CalSearchBar *cal_search)
priv = g_new (CalSearchBarPrivate, 1);
cal_search->priv = priv;
- priv->categories_omenu = NULL;
+ priv->categories = g_ptr_array_new ();
+ g_ptr_array_set_size (priv->categories, 0);
+}
+
+/* Frees an array of categories */
+static void
+free_categories (GPtrArray *categories)
+{
+ int i;
+
+ for (i = 0; i < categories->len; i++) {
+ g_assert (categories->pdata[i] != NULL);
+ g_free (categories->pdata[i]);
+ }
+
+ g_ptr_array_free (categories, TRUE);
}
/* Destroy handler for the calendar search bar */
@@ -183,7 +205,10 @@ cal_search_bar_destroy (GtkObject *object)
cal_search = CAL_SEARCH_BAR (object);
priv = cal_search->priv;
- priv->categories_omenu = NULL;
+ if (priv->categories) {
+ free_categories (priv->categories);
+ priv->categories = NULL;
+ }
g_free (priv);
cal_search->priv = NULL;
@@ -209,19 +234,47 @@ static const char *
get_current_category (CalSearchBar *cal_search)
{
CalSearchBarPrivate *priv;
- GtkMenu *menu;
- GtkWidget *active;
- const char *category;
+ int option, suboption;
priv = cal_search->priv;
- menu = GTK_MENU (gtk_option_menu_get_menu (priv->categories_omenu));
+ g_assert (priv->categories != NULL);
+
+ option = e_search_bar_get_option_choice (E_SEARCH_BAR (cal_search));
+ if (option != SEARCH_CATEGORY_IS)
+ return NULL;
+
+ suboption = e_search_bar_get_suboption_choice (E_SEARCH_BAR (cal_search));
+ if (suboption == CATEGORIES_ALL)
+ return (const char *) 1;
+ else if (suboption == CATEGORIES_UNMATCHED)
+ return NULL;
+ else {
+ int i;
+
+ i = suboption - CATEGORIES_OFFSET;
+ g_assert (i >= 0 && i < priv->categories->len);
+
+ return priv->categories->pdata[i];
+ }
+}
+
+/* Sets the query string to be (contains? "field" "text") */
+static void
+notify_query_contains (CalSearchBar *cal_search, const char *field)
+{
+ char *text;
+ char *sexp;
+
+ text = e_search_bar_get_text (E_SEARCH_BAR (cal_search));
+ if (!text)
+ return; /* This is an error in the UTF8 conversion, not an empty string! */
- active = gtk_menu_get_active (menu);
- g_assert (active != NULL);
+ sexp = g_strdup_printf ("(contains? \"%s\" \"%s\")", field, text);
+ g_free (text);
- category = gtk_object_get_user_data (GTK_OBJECT (active));
- return category;
+ notify_sexp_changed (cal_search, sexp);
+ g_free (sexp);
}
/* Returns a sexp for the selected category in the drop-down menu. The "All"
@@ -243,30 +296,20 @@ get_category_sexp (CalSearchBar *cal_search)
return g_strdup_printf ("(has-categories? \"%s\")", category); /* Specific category */
}
-/* Sets the query string to be (contains? "field" "text") */
+/* Sets the query string to the appropriate match for categories */
static void
-notify_query_contains (CalSearchBar *cal_search, const char *field, const char *text)
+notify_category_is (CalSearchBar *cal_search)
{
char *sexp;
- char *category_sexp;
-
- category_sexp = get_category_sexp (cal_search);
- if (category_sexp)
- /* "Contains" sexp plus a sexp for a category or for unfiled items */
- sexp = g_strdup_printf ("(and (contains? \"%s\" \"%s\")"
- " %s)",
- field, text, category_sexp);
+ sexp = get_category_sexp (cal_search);
+ if (!sexp)
+ notify_sexp_changed (cal_search, "#t"); /* Match all */
else
- /* "Contains" sexp; matches any category */
- sexp = g_strdup_printf ("(contains? \"%s\" \"%s\")", field, text);
+ notify_sexp_changed (cal_search, sexp);
- notify_sexp_changed (cal_search, sexp);
-
- if (category_sexp)
- g_free (category_sexp);
-
- g_free (sexp);
+ if (sexp)
+ g_free (sexp);
}
/* Creates a new query from the values in the widgets and notifies upstream */
@@ -275,42 +318,44 @@ regen_query (CalSearchBar *cal_search)
{
CalSearchBarPrivate *priv;
int item;
- char *text;
+ const char *category;
priv = cal_search->priv;
/* Fetch the data from the ESearchBar's entry widgets */
item = e_search_bar_get_option_choice (E_SEARCH_BAR (cal_search));
- text = e_search_bar_get_text (E_SEARCH_BAR (cal_search));
-
- if (!text)
- return; /* This is an error in the UTF8 conversion, not an empty string! */
/* Generate the different types of queries */
switch (item) {
case SEARCH_ANY_FIELD_CONTAINS:
- notify_query_contains (cal_search, "any", text);
+ notify_query_contains (cal_search, "any");
break;
case SEARCH_SUMMARY_CONTAINS:
- notify_query_contains (cal_search, "summary", text);
+ notify_query_contains (cal_search, "summary");
break;
case SEARCH_DESCRIPTION_CONTAINS:
- notify_query_contains (cal_search, "description", text);
+ notify_query_contains (cal_search, "description");
break;
case SEARCH_COMMENT_CONTAINS:
- notify_query_contains (cal_search, "comment", text);
+ notify_query_contains (cal_search, "comment");
+ break;
+
+ case SEARCH_CATEGORY_IS:
+ notify_category_is (cal_search);
+
+ category = cal_search_bar_get_category (cal_search);
+ gtk_signal_emit (GTK_OBJECT (cal_search), cal_search_bar_signals[CATEGORY_CHANGED],
+ category);
break;
default:
g_assert_not_reached ();
}
-
- g_free (text);
}
/* query_changed handler for the calendar search bar */
@@ -347,37 +392,59 @@ cal_search_bar_menu_activated (ESearchBar *search, int item)
-/* Callback used when an item is selected in the categories option menu */
+/* Creates the suboptions menu for the ESearchBar with the list of categories */
static void
-categories_selection_done_cb (GtkMenuShell *menu_shell, gpointer data)
+make_suboptions (CalSearchBar *cal_search)
{
- CalSearchBar *cal_search;
- const char *category;
+ CalSearchBarPrivate *priv;
+ ESearchBarSubitem *subitems;
+ int i;
- cal_search = CAL_SEARCH_BAR (data);
- regen_query (cal_search);
+ priv = cal_search->priv;
- category = cal_search_bar_get_category (cal_search);
- gtk_signal_emit (GTK_OBJECT (cal_search), cal_search_bar_signals[CATEGORY_CHANGED],
- category);
-}
+ g_assert (priv->categories != NULL);
-/* Creates the option menu of categories */
-static void
-setup_categories_omenu (CalSearchBar *cal_search)
-{
- CalSearchBarPrivate *priv;
- GtkWidget *label;
+ /* Categories plus "all", "unmatched", separator, terminator */
+ subitems = g_new (ESearchBarSubitem, priv->categories->len + 3 + 1);
- priv = cal_search->priv;
+ /* All, unmatched, separator */
+
+ subitems[0].text = _("Any");
+ subitems[0].id = CATEGORIES_ALL;
+ subitems[0].translate = FALSE;
+
+ subitems[1].text = _("Unmatched");
+ subitems[1].id = CATEGORIES_UNMATCHED;
+ subitems[1].translate = FALSE;
+
+ if (priv->categories->len > 0) {
+ subitems[2].text = NULL; /* separator */
+ subitems[2].id = 0;
+
+ /* All the other items */
- priv->categories_omenu = GTK_OPTION_MENU (gtk_option_menu_new ());
- gtk_box_pack_end (GTK_BOX (cal_search), GTK_WIDGET (priv->categories_omenu), FALSE, FALSE, 0);
- gtk_widget_show (GTK_WIDGET (priv->categories_omenu));
+ for (i = 0; i < priv->categories->len; i++) {
+ const char *category;
+ char *str;
- label = gtk_label_new (_("Category:"));
- gtk_box_pack_end (GTK_BOX (cal_search), label, FALSE, FALSE, 4);
- gtk_widget_show (label);
+ category = priv->categories->pdata[i];
+ str = e_utf8_to_gtk_string (GTK_WIDGET (cal_search), category);
+ if (!str)
+ str = g_strdup ("");
+
+ subitems[i + CATEGORIES_OFFSET].text = str;
+ subitems[i + CATEGORIES_OFFSET].id = i + CATEGORIES_OFFSET;
+ subitems[i + CATEGORIES_OFFSET].translate = FALSE;
+
+ g_free (str);
+ }
+
+ subitems[i + CATEGORIES_OFFSET].id = -1; /* terminator */
+ } else
+ subitems[2].id = -1; /* terminator */
+
+ e_search_bar_set_suboption (E_SEARCH_BAR (cal_search), SEARCH_CATEGORY_IS, subitems);
+ g_free (subitems);
}
/**
@@ -395,7 +462,7 @@ cal_search_bar_construct (CalSearchBar *cal_search)
g_return_val_if_fail (IS_CAL_SEARCH_BAR (cal_search), NULL);
e_search_bar_construct (E_SEARCH_BAR (cal_search), search_menu_items, search_option_items);
- setup_categories_omenu (cal_search);
+ make_suboptions (cal_search);
return cal_search;
}
@@ -417,20 +484,6 @@ cal_search_bar_new (void)
return GTK_WIDGET (cal_search_bar_construct (cal_search));
}
-/* Callback used when a categories menu item is destroyed. We free its user
- * data, which is the category string.
- */
-static void
-item_destroyed_cb (GtkObject *object, gpointer data)
-{
- char *category;
-
- category = gtk_object_get_user_data (object);
- g_assert (category != NULL);
-
- g_free (category);
-}
-
/* Used from qsort() */
static int
compare_categories_cb (const void *a, const void *b)
@@ -444,8 +497,8 @@ compare_categories_cb (const void *a, const void *b)
return strcmp (*ca, *cb);
}
-/* Creates a sorted array of categories based on the original one; does not
- * duplicate the string values.
+/* Creates a sorted array of categories based on the original one; copies the
+ * string values.
*/
static GPtrArray *
sort_categories (GPtrArray *categories)
@@ -457,7 +510,7 @@ sort_categories (GPtrArray *categories)
g_ptr_array_set_size (c, categories->len);
for (i = 0; i < categories->len; i++)
- c->pdata[i] = categories->pdata[i];
+ c->pdata[i] = g_strdup (categories->pdata[i]);
qsort (c->pdata, c->len, sizeof (gpointer), compare_categories_cb);
@@ -478,10 +531,6 @@ void
cal_search_bar_set_categories (CalSearchBar *cal_search, GPtrArray *categories)
{
CalSearchBarPrivate *priv;
- GtkMenu *menu;
- GtkWidget *item;
- GPtrArray *sorted;
- int i;
g_return_if_fail (cal_search != NULL);
g_return_if_fail (IS_CAL_SEARCH_BAR (cal_search));
@@ -489,59 +538,11 @@ cal_search_bar_set_categories (CalSearchBar *cal_search, GPtrArray *categories)
priv = cal_search->priv;
- menu = GTK_MENU (gtk_menu_new ());
- gtk_signal_connect (GTK_OBJECT (menu), "selection_done",
- GTK_SIGNAL_FUNC (categories_selection_done_cb), cal_search);
-
- /* All, Unmatched, separator items */
-
- item = gtk_menu_item_new_with_label (_("All"));
- gtk_object_set_user_data (GTK_OBJECT (item), (char *) 1);
- gtk_menu_append (menu, item);
- gtk_widget_show (item);
-
- item = gtk_menu_item_new_with_label (_("Unfiled"));
- gtk_object_set_user_data (GTK_OBJECT (item), NULL);
- gtk_menu_append (menu, item);
- gtk_widget_show (item);
-
- if (categories->len > 0) {
- item = gtk_menu_item_new ();
- gtk_widget_set_sensitive (item, FALSE);
- gtk_menu_append (menu, item);
- gtk_widget_show (item);
- }
-
- /* Categories items */
-
- sorted = sort_categories (categories);
-
- for (i = 0; i < sorted->len; i++) {
- char *str;
-
- /* FIXME: Put the category icons here */
-
- str = e_utf8_to_gtk_string (GTK_WIDGET (menu), sorted->pdata[i]);
- if (!str)
- continue;
-
- item = gtk_menu_item_new_with_label (str);
- g_free (str);
-
- gtk_object_set_user_data (GTK_OBJECT (item), g_strdup (sorted->pdata[i]));
- gtk_signal_connect (GTK_OBJECT (item), "destroy",
- GTK_SIGNAL_FUNC (item_destroyed_cb),
- NULL);
-
- gtk_menu_append (menu, item);
- gtk_widget_show (item);
- }
-
- g_ptr_array_free (sorted, TRUE);
-
- /* Set the new menu; the old one will be destroyed automatically */
+ g_assert (priv->categories != NULL);
+ free_categories (priv->categories);
- gtk_option_menu_set_menu (priv->categories_omenu, GTK_WIDGET (menu));
+ priv->categories = sort_categories (categories);
+ make_suboptions (cal_search);
}
/**