* @file gtkstatusbox.c GTK+ Status Selection Widget * Gaim is the legal property of its developers, whose names are too numerous * to list here. Please refer to the COPYRIGHT file distributed with this * 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 * The status box is made up of two main pieces: * - The box that displays the current status, which is made * of a GtkListStore ("status_box->store") and GtkCellView * ("status_box->cell_view"). There is always exactly 1 row * in this list store. Only the TYPE_ICON and TYPE_TEXT * columns are used in this list store. * - The dropdown menu that lets users select a status, which * is made of a GtkComboBox ("status_box") and GtkListStore * ("status_box->dropdown_store"). This dropdown is shown * when the user clicks on the box that displays the current * status. This list store contains one row for Available, * one row for Away, etc., a few rows for popular statuses, * and the "New..." and "Saved..." options. #include <gdk/gdkkeysyms.h> #include "savedstatuses.h" #include "gtksavedstatuses.h" #include "gtkstatusbox.h" # include <gtkspell/gtkspell.h> #define TYPING_TIMEOUT 4000 static void imhtml_changed_cb(GtkTextBuffer *buffer, void *data); static void imhtml_format_changed_cb(GtkIMHtml *imhtml, GtkIMHtmlButtons buttons, void *data); static void remove_typing_cb(GtkGaimStatusBox *box); static void update_size (GtkGaimStatusBox *box); static gint get_statusbox_index(GtkGaimStatusBox *box, GaimSavedStatus *saved_status); static void gtk_gaim_status_box_pulse_typing(GtkGaimStatusBox *status_box); static void gtk_gaim_status_box_refresh(GtkGaimStatusBox *status_box); static void gtk_gaim_status_box_regenerate(GtkGaimStatusBox *status_box); static void gtk_gaim_status_box_changed(GtkComboBox *box); static void gtk_gaim_status_box_size_request (GtkWidget *widget, GtkRequisition *requisition); static void gtk_gaim_status_box_size_allocate (GtkWidget *widget, GtkAllocation *allocation); static gboolean gtk_gaim_status_box_expose_event (GtkWidget *widget, GdkEventExpose *event); static void gtk_gaim_status_box_forall (GtkContainer *container, gboolean include_internals, GtkCallback callback, gpointer callback_data); static void (*combo_box_size_request)(GtkWidget *widget, GtkRequisition *requisition); static void (*combo_box_size_allocate)(GtkWidget *widget, GtkAllocation *allocation); static void (*combo_box_forall) (GtkContainer *container, gboolean include_internals, GtkCallback callback, gpointer callback_data); /** A GtkGaimStatusBoxItemType */ * This is a GdkPixbuf (the other columns are strings). * This column is visible. /** The text displayed on the status box. This column is visible. */ /** The plain-English title of this item */ /** A plain-English description of this item */ * This value depends on TYPE_COLUMN. For POPULAR types, * this is the creation time. For PRIMITIVE types, * this is the GaimStatusPrimitive. GtkComboBoxClass *parent_class = NULL; static void gtk_gaim_status_box_class_init (GtkGaimStatusBoxClass *klass); static void gtk_gaim_status_box_init (GtkGaimStatusBox *status_box); gtk_gaim_status_box_get_type (void) static GType status_box_type = 0; static const GTypeInfo status_box_info = sizeof (GtkGaimStatusBoxClass), NULL, /* base_finalize */ (GClassInitFunc) gtk_gaim_status_box_class_init, NULL, /* class_finalize */ sizeof (GtkGaimStatusBox), (GInstanceInitFunc) gtk_gaim_status_box_init, status_box_type = g_type_register_static(GTK_TYPE_COMBO_BOX, gtk_gaim_status_box_get_property(GObject *object, guint param_id, GValue *value, GParamSpec *psec) GtkGaimStatusBox *statusbox = GTK_GAIM_STATUS_BOX(object); g_value_set_pointer(value, statusbox->account); G_OBJECT_WARN_INVALID_PROPERTY_ID(object, param_id, psec); update_to_reflect_account_status(GtkGaimStatusBox *status_box, GaimAccount *account, GaimStatus *newstatus) const GaimStatusType *statustype = NULL; statustype = gaim_status_type_find_with_id((GList *)gaim_account_get_status_types(account), (char *)gaim_status_type_get_id(gaim_status_get_type(newstatus))); for (l = gaim_account_get_status_types(account); l != NULL; l = l->next) { GaimStatusType *status_type = (GaimStatusType *)l->data; if (!gaim_status_type_is_user_settable(status_type)) if (statustype == status_type) gtk_widget_set_sensitive(GTK_WIDGET(status_box), FALSE); gtk_combo_box_set_active(GTK_COMBO_BOX(status_box), status_no); message = gaim_status_get_attr_string(newstatus, "message"); if (!message || !*message) gtk_widget_hide_all(status_box->vbox); status_box->imhtml_visible = FALSE; gtk_widget_show_all(status_box->vbox); status_box->imhtml_visible = TRUE; gtk_imhtml_clear(GTK_IMHTML(status_box->imhtml)); gtk_imhtml_clear_formatting(GTK_IMHTML(status_box->imhtml)); gtk_imhtml_append_text(GTK_IMHTML(status_box->imhtml), message, 0); gtk_widget_set_sensitive(GTK_WIDGET(status_box), TRUE); gtk_gaim_status_box_refresh(status_box); account_status_changed_cb(GaimAccount *account, GaimStatus *oldstatus, GaimStatus *newstatus, GtkGaimStatusBox *status_box) if (status_box->account == account) update_to_reflect_account_status(status_box, account, newstatus); gtk_gaim_status_box_set_property(GObject *object, guint param_id, const GValue *value, GParamSpec *pspec) GtkGaimStatusBox *statusbox = GTK_GAIM_STATUS_BOX(object); statusbox->account = g_value_get_pointer(value); if (statusbox->status_changed_signal) { gaim_signal_disconnect(gaim_accounts_get_handle(), "account-status-changed", statusbox, GAIM_CALLBACK(account_status_changed_cb)); statusbox->status_changed_signal = 0; statusbox->status_changed_signal = gaim_signal_connect(gaim_accounts_get_handle(), "account-status-changed", statusbox, GAIM_CALLBACK(account_status_changed_cb), gtk_gaim_status_box_regenerate(statusbox); G_OBJECT_WARN_INVALID_PROPERTY_ID(object, param_id, pspec); gtk_gaim_status_box_finalize(GObject *obj) GtkGaimStatusBox *statusbox = GTK_GAIM_STATUS_BOX(obj); if (statusbox->status_changed_signal) { gaim_signal_disconnect(gaim_accounts_get_handle(), "account-status-changed", statusbox, GAIM_CALLBACK(account_status_changed_cb)); statusbox->status_changed_signal = 0; gaim_prefs_disconnect_by_handle(statusbox); G_OBJECT_CLASS(parent_class)->finalize(obj); gtk_gaim_status_box_class_init (GtkGaimStatusBoxClass *klass) GObjectClass *object_class; GtkComboBoxClass *combo_class; GtkWidgetClass *widget_class; GtkContainerClass *container_class = (GtkContainerClass*)klass; parent_class = g_type_class_peek_parent(klass); combo_class = (GtkComboBoxClass*)klass; combo_class->changed = gtk_gaim_status_box_changed; widget_class = (GtkWidgetClass*)klass; combo_box_size_request = widget_class->size_request; widget_class->size_request = gtk_gaim_status_box_size_request; combo_box_size_allocate = widget_class->size_allocate; widget_class->size_allocate = gtk_gaim_status_box_size_allocate; widget_class->expose_event = gtk_gaim_status_box_expose_event; combo_box_forall = container_class->forall; container_class->forall = gtk_gaim_status_box_forall; object_class = (GObjectClass *)klass; object_class->finalize = gtk_gaim_status_box_finalize; object_class->get_property = gtk_gaim_status_box_get_property; object_class->set_property = gtk_gaim_status_box_set_property; g_object_class_install_property(object_class, g_param_spec_pointer("account", "The account, or NULL for all accounts", * This updates the text displayed on the status box so that it shows * the current status. This is the only function in this file that * should modify status_box->store gtk_gaim_status_box_refresh(GtkGaimStatusBox *status_box) gboolean show_buddy_icons; GaimSavedStatus *saved_status; char *primary, *secondary, *text; show_buddy_icons = gaim_prefs_get_bool("/gaim/gtk/blist/show_buddy_icons"); icon_size = gtk_icon_size_from_name(GAIM_ICON_SIZE_STATUS); icon_size = gtk_icon_size_from_name(GAIM_ICON_SIZE_STATUS_SMALL); style = gtk_widget_get_style(GTK_WIDGET(status_box)); snprintf(aa_color, sizeof(aa_color), "#%02x%02x%02x", style->text_aa[GTK_STATE_NORMAL].red >> 8, style->text_aa[GTK_STATE_NORMAL].green >> 8, style->text_aa[GTK_STATE_NORMAL].blue >> 8); saved_status = gaim_savedstatus_get_current(); if (status_box->typing != 0) GtkGaimStatusBoxItemType type; /* Primary (get the status selected in the dropdown) */ gtk_combo_box_get_active_iter(GTK_COMBO_BOX(status_box), &iter); gtk_tree_model_get(GTK_TREE_MODEL(status_box->dropdown_store), &iter, if (type == GTK_GAIM_STATUS_BOX_TYPE_PRIMITIVE) primary = g_strdup(gaim_primitive_get_name_from_type(GPOINTER_TO_INT(data))); /* This should never happen, but just in case... */ primary = g_strdup("New status"); else if (status_box->account != NULL) primary = g_strdup(gaim_status_get_name(gaim_account_get_active_status(status_box->account))); else if (gaim_savedstatus_is_transient(saved_status)) primary = g_strdup(gaim_primitive_get_name_from_type(gaim_savedstatus_get_type(saved_status))); primary = g_markup_escape_text(gaim_savedstatus_get_title(saved_status), -1); if (status_box->typing != 0) secondary = g_strdup(_("Typing")); else if (status_box->connecting) secondary = g_strdup(_("Connecting")); else if (gaim_savedstatus_is_transient(saved_status)) message = gaim_savedstatus_get_message(saved_status); tmp = gaim_markup_strip_html(message); gaim_util_chrreplace(tmp, '\n', ' '); secondary = g_markup_escape_text(tmp, -1); if (status_box->typing != 0) pixbuf = status_box->typing_pixbufs[status_box->typing_index]; else if (status_box->connecting) pixbuf = status_box->connecting_pixbufs[status_box->connecting_index]; if (status_box->account != NULL) pixbuf = gaim_gtk_create_prpl_icon_with_status(status_box->account, gaim_status_get_type(gaim_account_get_active_status(status_box->account)), show_buddy_icons ? 1.0 : 0.5); pixbuf = gaim_gtk_create_gaim_icon_with_status( gaim_savedstatus_get_type(saved_status), show_buddy_icons ? 1.0 : 0.5); if (!gaim_savedstatus_is_transient(saved_status)) /* Overlay a disk in the bottom left corner */ emblem = gtk_widget_render_icon(GTK_WIDGET(status_box->vbox), GTK_STOCK_SAVE, icon_size, "GtkGaimStatusBox"); width = gdk_pixbuf_get_width(pixbuf) / 2; height = gdk_pixbuf_get_height(pixbuf) / 2; gdk_pixbuf_composite(emblem, pixbuf, 0, height, width, height, 0, height, 0.5, 0.5, GDK_INTERP_BILINEAR, 255); g_object_unref(G_OBJECT(emblem)); if (status_box->account != NULL) { text = g_strdup_printf("%s%s<span size=\"smaller\" color=\"%s\">%s</span>", gaim_account_get_username(status_box->account), show_buddy_icons ? "\n" : " - ", aa_color, secondary ? secondary : primary); } else if (secondary != NULL) { separator = show_buddy_icons ? "\n" : " - "; text = g_strdup_printf("%s<span size=\"smaller\" color=\"%s\">%s%s</span>", primary, aa_color, separator, secondary); text = g_strdup(primary); * Only two columns are used in this list store (does it * really need to be a list store?) gtk_list_store_set(status_box->store, &(status_box->iter), if ((status_box->typing == 0) && (!status_box->connecting)) /* Make sure to activate the only row in the tree view */ path = gtk_tree_path_new_from_string("0"); gtk_cell_view_set_displayed_row(GTK_CELL_VIEW(status_box->cell_view), path); gtk_tree_path_free(path); * This updates the GtkTreeView so that it correctly shows the state * we are currently using. It is used when the current state is * updated from somewhere other than the GtkStatusBox (from a plugin, * or when signing on with the "-n" option, for example). It is * also used when the user selects the "New..." option. * Maybe we could accomplish this by triggering off the mouse and * keyboard signals instead of the changed signal? status_menu_refresh_iter(GtkGaimStatusBox *status_box) GaimSavedStatus *saved_status; GaimStatusPrimitive primitive; /* this function is inappropriate for ones with accounts */ saved_status = gaim_savedstatus_get_current(); * Suppress the "changed" signal because the status * was changed programmatically. gtk_widget_set_sensitive(GTK_WIDGET(status_box), FALSE); * If the saved_status is transient, is Available, Away, Invisible * or Offline, and it does not have an substatuses, then select * the primitive in the dropdown menu. Otherwise select the * popular status in the dropdown menu. primitive = gaim_savedstatus_get_type(saved_status); if (gaim_savedstatus_is_transient(saved_status) && ((primitive == GAIM_STATUS_AVAILABLE) || (primitive == GAIM_STATUS_AWAY) || (primitive == GAIM_STATUS_INVISIBLE) || (primitive == GAIM_STATUS_OFFLINE)) && (!gaim_savedstatus_has_substatuses(saved_status))) index = get_statusbox_index(status_box, saved_status); gtk_combo_box_set_active(GTK_COMBO_BOX(status_box), index); GtkGaimStatusBoxItemType type; /* Unset the active item */ gtk_combo_box_set_active(GTK_COMBO_BOX(status_box), -1); /* If this saved status is in the list store, then set it as the active item */ if (gtk_tree_model_get_iter_first(GTK_TREE_MODEL(status_box->dropdown_store), &iter)) gtk_tree_model_get(GTK_TREE_MODEL(status_box->dropdown_store), &iter, if ((type == GTK_GAIM_STATUS_BOX_TYPE_POPULAR) && (GPOINTER_TO_INT(data) == gaim_savedstatus_get_creation_time(saved_status))) gtk_combo_box_set_active_iter(GTK_COMBO_BOX(status_box), &iter); while (gtk_tree_model_iter_next(GTK_TREE_MODEL(status_box->dropdown_store), &iter)); message = gaim_savedstatus_get_message(saved_status); if (!gaim_savedstatus_is_transient(saved_status) || !message || !*message) status_box->imhtml_visible = FALSE; gtk_widget_hide_all(status_box->vbox); status_box->imhtml_visible = TRUE; gtk_widget_show_all(status_box->vbox); * Suppress the "changed" signal because the status * was changed programmatically. gtk_widget_set_sensitive(GTK_WIDGET(status_box->imhtml), FALSE); gtk_imhtml_clear(GTK_IMHTML(status_box->imhtml)); gtk_imhtml_clear_formatting(GTK_IMHTML(status_box->imhtml)); gtk_imhtml_append_text(GTK_IMHTML(status_box->imhtml), message, 0); gtk_widget_set_sensitive(GTK_WIDGET(status_box->imhtml), TRUE); /* Stop suppressing the "changed" signal. */ gtk_widget_set_sensitive(GTK_WIDGET(status_box), TRUE); add_popular_statuses(GtkGaimStatusBox *statusbox) gboolean show_buddy_icons; GdkPixbuf *pixbuf, *emblem; list = gaim_savedstatuses_get_popular(6); /* Odd... oh well, nothing we can do about it. */ show_buddy_icons = gaim_prefs_get_bool("/gaim/gtk/blist/show_buddy_icons"); icon_size = gtk_icon_size_from_name(GAIM_ICON_SIZE_STATUS); icon_size = gtk_icon_size_from_name(GAIM_ICON_SIZE_STATUS_SMALL); gtk_gaim_status_box_add_separator(statusbox); for (cur = list; cur != NULL; cur = cur->next) GaimSavedStatus *saved = cur->data; /* Get an appropriate status icon */ pixbuf = gaim_gtk_create_gaim_icon_with_status( gaim_savedstatus_get_type(saved), show_buddy_icons ? 1.0 : 0.5); if (gaim_savedstatus_is_transient(saved)) * Transient statuses do not have a title, so the savedstatus * API returns the message when gaim_savedstatus_get_title() is * called, so we don't need to get the message a second time. message = gaim_savedstatus_get_message(saved); stripped = gaim_markup_strip_html(message); gaim_util_chrreplace(stripped, '\n', ' '); /* Overlay a disk in the bottom left corner */ emblem = gtk_widget_render_icon(GTK_WIDGET(statusbox->vbox), GTK_STOCK_SAVE, icon_size, "GtkGaimStatusBox"); width = gdk_pixbuf_get_width(pixbuf) / 2; height = gdk_pixbuf_get_height(pixbuf) / 2; gdk_pixbuf_composite(emblem, pixbuf, 0, height, width, height, 0, height, 0.5, 0.5, GDK_INTERP_BILINEAR, 255); g_object_unref(G_OBJECT(emblem)); gtk_gaim_status_box_add(statusbox, GTK_GAIM_STATUS_BOX_TYPE_POPULAR, pixbuf, gaim_savedstatus_get_title(saved), stripped, GINT_TO_POINTER(gaim_savedstatus_get_creation_time(saved))); g_object_unref(G_OBJECT(pixbuf)); gtk_gaim_status_box_regenerate(GtkGaimStatusBox *status_box) gboolean show_buddy_icons; GdkPixbuf *pixbuf, *pixbuf2, *pixbuf3, *pixbuf4, *tmp; show_buddy_icons = gaim_prefs_get_bool("/gaim/gtk/blist/show_buddy_icons"); icon_size = gtk_icon_size_from_name(GAIM_ICON_SIZE_STATUS); icon_size = gtk_icon_size_from_name(GAIM_ICON_SIZE_STATUS_SMALL); /* Unset the model while clearing it */ gtk_combo_box_set_model(GTK_COMBO_BOX(status_box), NULL); gtk_list_store_clear(status_box->dropdown_store); /* Don't set the model until the new statuses have been added to the box. * What is presumably a bug in Gtk < 2.4 causes things to get all confused /* gtk_combo_box_set_model(GTK_COMBO_BOX(status_box), GTK_TREE_MODEL(status_box->dropdown_store)); */ account = GTK_GAIM_STATUS_BOX(status_box)->account; pixbuf = gtk_widget_render_icon (GTK_WIDGET(status_box->vbox), GAIM_STOCK_STATUS_ONLINE, icon_size, "GtkGaimStatusBox"); pixbuf2 = gtk_widget_render_icon (GTK_WIDGET(status_box->vbox), GAIM_STOCK_STATUS_AWAY, icon_size, "GtkGaimStatusBox"); pixbuf3 = gtk_widget_render_icon (GTK_WIDGET(status_box->vbox), GAIM_STOCK_STATUS_OFFLINE, icon_size, "GtkGaimStatusBox"); pixbuf4 = gtk_widget_render_icon (GTK_WIDGET(status_box->vbox), GAIM_STOCK_STATUS_INVISIBLE, icon_size, "GtkGaimStatusBox"); gtk_gaim_status_box_add(GTK_GAIM_STATUS_BOX(status_box), GTK_GAIM_STATUS_BOX_TYPE_PRIMITIVE, pixbuf, _("Available"), NULL, GINT_TO_POINTER(GAIM_STATUS_AVAILABLE)); gtk_gaim_status_box_add(GTK_GAIM_STATUS_BOX(status_box), GTK_GAIM_STATUS_BOX_TYPE_PRIMITIVE, pixbuf2, _("Away"), NULL, GINT_TO_POINTER(GAIM_STATUS_AWAY)); gtk_gaim_status_box_add(GTK_GAIM_STATUS_BOX(status_box), GTK_GAIM_STATUS_BOX_TYPE_PRIMITIVE, pixbuf4, _("Invisible"), NULL, GINT_TO_POINTER(GAIM_STATUS_INVISIBLE)); gtk_gaim_status_box_add(GTK_GAIM_STATUS_BOX(status_box), GTK_GAIM_STATUS_BOX_TYPE_PRIMITIVE, pixbuf3, _("Offline"), NULL, GINT_TO_POINTER(GAIM_STATUS_OFFLINE)); add_popular_statuses(status_box); gtk_gaim_status_box_add_separator(GTK_GAIM_STATUS_BOX(status_box)); gtk_gaim_status_box_add(GTK_GAIM_STATUS_BOX(status_box), GTK_GAIM_STATUS_BOX_TYPE_CUSTOM, pixbuf, _("New..."), NULL, NULL); gtk_gaim_status_box_add(GTK_GAIM_STATUS_BOX(status_box), GTK_GAIM_STATUS_BOX_TYPE_SAVED, pixbuf, _("Saved..."), NULL, NULL); gtk_combo_box_set_model(GTK_COMBO_BOX(status_box), GTK_TREE_MODEL(status_box->dropdown_store)); status_menu_refresh_iter(status_box); for (l = gaim_account_get_status_types(account); l != NULL; l = l->next) GaimStatusType *status_type = (GaimStatusType *)l->data; if (!gaim_status_type_is_user_settable(status_type)) tmp = gaim_gtk_create_prpl_icon_with_status(account, status_type, show_buddy_icons ? 1.0 : 0.5); gtk_gaim_status_box_add(GTK_GAIM_STATUS_BOX(status_box), GTK_GAIM_STATUS_BOX_TYPE_PRIMITIVE, tmp, gaim_status_type_get_name(status_type), GINT_TO_POINTER(gaim_status_type_get_primitive(status_type))); gtk_combo_box_set_model(GTK_COMBO_BOX(status_box), GTK_TREE_MODEL(status_box->dropdown_store)); update_to_reflect_account_status(status_box, account, gaim_account_get_active_status(account)); static gboolean combo_box_scroll_event_cb(GtkWidget *w, GdkEventScroll *event, GtkIMHtml *imhtml) gtk_combo_box_popup(GTK_COMBO_BOX(w)); static gboolean imhtml_scroll_event_cb(GtkWidget *w, GdkEventScroll *event, GtkIMHtml *imhtml) if (event->direction == GDK_SCROLL_UP) gtk_imhtml_page_up(imhtml); else if (event->direction == GDK_SCROLL_DOWN) gtk_imhtml_page_down(imhtml); static int imhtml_remove_focus(GtkWidget *w, GdkEventKey *event, GtkGaimStatusBox *status_box) if (event->keyval == GDK_Tab || event->keyval == GDK_KP_Tab) /* If last inserted character is a tab, then remove the focus from here */ GtkWidget *top = gtk_widget_get_toplevel(w); g_signal_emit_by_name(G_OBJECT(top), "move_focus", (event->state & GDK_SHIFT_MASK) ? GTK_DIR_TAB_BACKWARD: GTK_DIR_TAB_FORWARD); if (!status_box->typing != 0) gtk_gaim_status_box_pulse_typing(status_box); g_source_remove(status_box->typing); status_box->typing = g_timeout_add(TYPING_TIMEOUT, (GSourceFunc)remove_typing_cb, status_box); #if GTK_CHECK_VERSION(2,6,0) dropdown_store_row_separator_func(GtkTreeModel *model, GtkTreeIter *iter, gpointer data) GtkGaimStatusBoxItemType type; gtk_tree_model_get(model, iter, TYPE_COLUMN, &type, -1); if (type == GTK_GAIM_STATUS_BOX_TYPE_SEPARATOR) cache_pixbufs(GtkGaimStatusBox *status_box) if (gaim_prefs_get_bool("/gaim/gtk/blist/show_buddy_icons")) g_object_set(G_OBJECT(status_box->icon_rend), "xpad", 6, NULL); icon_size = gtk_icon_size_from_name(GAIM_ICON_SIZE_STATUS_TWO_LINE); g_object_set(G_OBJECT(status_box->icon_rend), "xpad", 3, NULL); icon_size = gtk_icon_size_from_name(GAIM_ICON_SIZE_STATUS_SMALL_TWO_LINE); if (status_box->connecting_pixbufs[0] != NULL) gdk_pixbuf_unref(status_box->connecting_pixbufs[0]); if (status_box->connecting_pixbufs[1] != NULL) gdk_pixbuf_unref(status_box->connecting_pixbufs[1]); if (status_box->connecting_pixbufs[2] != NULL) gdk_pixbuf_unref(status_box->connecting_pixbufs[2]); if (status_box->connecting_pixbufs[3] != NULL) gdk_pixbuf_unref(status_box->connecting_pixbufs[3]); status_box->connecting_index = 0; status_box->connecting_pixbufs[0] = gtk_widget_render_icon (GTK_WIDGET(status_box->vbox), GAIM_STOCK_STATUS_CONNECT0, icon_size, "GtkGaimStatusBox"); status_box->connecting_pixbufs[1] = gtk_widget_render_icon (GTK_WIDGET(status_box->vbox), GAIM_STOCK_STATUS_CONNECT1, icon_size, "GtkGaimStatusBox"); status_box->connecting_pixbufs[2] = gtk_widget_render_icon (GTK_WIDGET(status_box->vbox), GAIM_STOCK_STATUS_CONNECT2, icon_size, "GtkGaimStatusBox"); status_box->connecting_pixbufs[3] = gtk_widget_render_icon (GTK_WIDGET(status_box->vbox), GAIM_STOCK_STATUS_CONNECT3, icon_size, "GtkGaimStatusBox"); if (status_box->typing_pixbufs[0] != NULL) gdk_pixbuf_unref(status_box->typing_pixbufs[0]); if (status_box->typing_pixbufs[1] != NULL) gdk_pixbuf_unref(status_box->typing_pixbufs[1]); if (status_box->typing_pixbufs[2] != NULL) gdk_pixbuf_unref(status_box->typing_pixbufs[2]); if (status_box->typing_pixbufs[3] != NULL) gdk_pixbuf_unref(status_box->typing_pixbufs[3]); status_box->typing_index = 0; status_box->typing_pixbufs[0] = gtk_widget_render_icon (GTK_WIDGET(status_box->vbox), GAIM_STOCK_STATUS_TYPING0, icon_size, "GtkGaimStatusBox"); status_box->typing_pixbufs[1] = gtk_widget_render_icon (GTK_WIDGET(status_box->vbox), GAIM_STOCK_STATUS_TYPING1, icon_size, "GtkGaimStatusBox"); status_box->typing_pixbufs[2] = gtk_widget_render_icon (GTK_WIDGET(status_box->vbox), GAIM_STOCK_STATUS_TYPING2, icon_size, "GtkGaimStatusBox"); status_box->typing_pixbufs[3] = gtk_widget_render_icon (GTK_WIDGET(status_box->vbox), GAIM_STOCK_STATUS_TYPING3, icon_size, "GtkGaimStatusBox"); current_status_pref_changed_cb(const char *name, GaimPrefType type, gconstpointer val, gpointer data) GtkGaimStatusBox *status_box = data; /* Make sure our current status is added to the list of popular statuses */ gtk_gaim_status_box_regenerate(status_box); if (status_box->account != NULL) update_to_reflect_account_status(status_box, status_box->account, gaim_account_get_active_status(status_box->account)); status_menu_refresh_iter(status_box); gtk_gaim_status_box_refresh(status_box); buddy_list_details_pref_changed_cb(const char *name, GaimPrefType type, gconstpointer val, gpointer data) GtkGaimStatusBox *status_box = (GtkGaimStatusBox *)data; cache_pixbufs(status_box); gtk_gaim_status_box_regenerate(status_box); gtk_gaim_status_box_refresh(status_box); spellcheck_prefs_cb(const char *name, GaimPrefType type, gconstpointer value, gpointer data) GtkGaimStatusBox *status_box = (GtkGaimStatusBox *)data; gaim_gtk_setup_gtkspell(GTK_TEXT_VIEW(status_box->imhtml)); spell = gtkspell_get_from_text_view(GTK_TEXT_VIEW(status_box->imhtml)); static gboolean button_released_cb(GtkWidget *widget, GdkEventButton *event, GtkGaimStatusBox *box) gtk_combo_box_popdown(GTK_COMBO_BOX(box)); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(box->toggle_button), FALSE); if (!box->imhtml_visible) g_signal_emit_by_name(G_OBJECT(box), "changed", NULL, NULL); static gboolean button_pressed_cb(GtkWidget *widget, GdkEventButton *event, GtkGaimStatusBox *box) gtk_combo_box_popup(GTK_COMBO_BOX(box)); // Disabled until button_released_cb works // gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(box->toggle_button), TRUE); toggled_cb(GtkWidget *widget, GtkGaimStatusBox *box) gtk_combo_box_popup(GTK_COMBO_BOX(box)); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(box->toggle_button), FALSE); gtk_gaim_status_box_init (GtkGaimStatusBox *status_box) GtkCellRenderer *text_rend; GtkCellRenderer *icon_rend; status_box->imhtml_visible = FALSE; status_box->connecting = FALSE; status_box->toggle_button = gtk_toggle_button_new(); status_box->hbox = gtk_hbox_new(FALSE, 6); status_box->cell_view = gtk_cell_view_new(); status_box->vsep = gtk_vseparator_new(); status_box->arrow = gtk_arrow_new (GTK_ARROW_DOWN, GTK_SHADOW_NONE); status_box->store = gtk_list_store_new(NUM_COLUMNS, G_TYPE_INT, GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_POINTER); status_box->dropdown_store = gtk_list_store_new(NUM_COLUMNS, G_TYPE_INT, GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_POINTER); gtk_combo_box_set_model(GTK_COMBO_BOX(status_box), GTK_TREE_MODEL(status_box->dropdown_store)); gtk_cell_view_set_model(GTK_CELL_VIEW(status_box->cell_view), GTK_TREE_MODEL(status_box->store)); gtk_combo_box_set_wrap_width(GTK_COMBO_BOX(status_box), 0); gtk_list_store_append(status_box->store, &(status_box->iter)); gtk_gaim_status_box_refresh(status_box); gtk_container_add(GTK_CONTAINER(status_box->toggle_button), status_box->hbox); gtk_box_pack_start(GTK_BOX(status_box->hbox), status_box->cell_view, TRUE, TRUE, 0); gtk_box_pack_start(GTK_BOX(status_box->hbox), status_box->vsep, FALSE, FALSE, 0); gtk_box_pack_start(GTK_BOX(status_box->hbox), status_box->arrow, FALSE, FALSE, 0); gtk_widget_show_all(status_box->toggle_button); #if GTK_CHECK_VERSION(2,4,0) gtk_button_set_focus_on_click(GTK_BUTTON(status_box->toggle_button), FALSE); text_rend = gtk_cell_renderer_text_new(); icon_rend = gtk_cell_renderer_pixbuf_new(); gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(status_box), icon_rend, FALSE); gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(status_box), text_rend, TRUE); gtk_cell_layout_set_attributes(GTK_CELL_LAYOUT(status_box), icon_rend, "pixbuf", ICON_COLUMN, NULL); gtk_cell_layout_set_attributes(GTK_CELL_LAYOUT(status_box), text_rend, "markup", TEXT_COLUMN, NULL); #if GTK_CHECK_VERSION(2, 6, 0) g_object_set(text_rend, "ellipsize", PANGO_ELLIPSIZE_END, NULL); status_box->icon_rend = gtk_cell_renderer_pixbuf_new(); status_box->text_rend = gtk_cell_renderer_text_new(); gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(status_box->cell_view), status_box->icon_rend, FALSE); gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(status_box->cell_view), status_box->text_rend, TRUE); gtk_cell_layout_set_attributes(GTK_CELL_LAYOUT(status_box->cell_view), status_box->icon_rend, "pixbuf", ICON_COLUMN, NULL); gtk_cell_layout_set_attributes(GTK_CELL_LAYOUT(status_box->cell_view), status_box->text_rend, "markup", TEXT_COLUMN, NULL); #if GTK_CHECK_VERSION(2, 6, 0) g_object_set(status_box->text_rend, "ellipsize", PANGO_ELLIPSIZE_END, NULL); status_box->vbox = gtk_vbox_new(0, FALSE); status_box->sw = gaim_gtk_create_imhtml(FALSE, &status_box->imhtml, NULL, NULL); gtk_imhtml_set_editable(GTK_IMHTML(status_box->imhtml), TRUE); buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(status_box->imhtml)); g_signal_connect(G_OBJECT(status_box->toggle_button), "button-press-event", G_CALLBACK(button_pressed_cb), status_box); g_signal_connect(G_OBJECT(status_box->toggle_button), "button-release-event", G_CALLBACK(button_released_cb), status_box); g_signal_connect(G_OBJECT(status_box->toggle_button), "toggled", G_CALLBACK(toggled_cb), status_box); g_signal_connect(G_OBJECT(buffer), "changed", G_CALLBACK(imhtml_changed_cb), status_box); g_signal_connect(G_OBJECT(status_box->imhtml), "format_function_toggle", G_CALLBACK(imhtml_format_changed_cb), status_box); g_signal_connect(G_OBJECT(status_box->imhtml), "key_press_event", G_CALLBACK(imhtml_remove_focus), status_box); g_signal_connect_swapped(G_OBJECT(status_box->imhtml), "message_send", G_CALLBACK(remove_typing_cb), status_box); gtk_imhtml_set_editable(GTK_IMHTML(status_box->imhtml), TRUE); if (gaim_prefs_get_bool("/gaim/gtk/conversations/spellcheck")) gaim_gtk_setup_gtkspell(GTK_TEXT_VIEW(status_box->imhtml)); gtk_widget_set_parent(status_box->vbox, GTK_WIDGET(status_box)); gtk_widget_set_parent(status_box->toggle_button, GTK_WIDGET(status_box)); GTK_BIN(status_box)->child = status_box->toggle_button; gtk_box_pack_start(GTK_BOX(status_box->vbox), status_box->sw, TRUE, TRUE, 0); g_signal_connect(G_OBJECT(status_box), "scroll_event", G_CALLBACK(combo_box_scroll_event_cb), NULL); g_signal_connect(G_OBJECT(status_box->imhtml), "scroll_event", G_CALLBACK(imhtml_scroll_event_cb), status_box->imhtml); #if GTK_CHECK_VERSION(2,6,0) gtk_combo_box_set_row_separator_func(GTK_COMBO_BOX(status_box), dropdown_store_row_separator_func, NULL, NULL); cache_pixbufs(status_box); gtk_gaim_status_box_regenerate(status_box); gaim_prefs_connect_callback(status_box, "/core/savedstatus/current", current_status_pref_changed_cb, status_box); gaim_prefs_connect_callback(status_box, "/gaim/gtk/blist/show_buddy_icons", buddy_list_details_pref_changed_cb, status_box); gaim_prefs_connect_callback(status_box, "/gaim/gtk/conversations/spellcheck", spellcheck_prefs_cb, status_box); gtk_gaim_status_box_size_request(GtkWidget *widget, GtkRequisition *requisition) combo_box_size_request(widget, requisition); requisition->height += 3; /* If the gtkimhtml is visible, then add some additional padding */ gtk_widget_size_request(GTK_GAIM_STATUS_BOX(widget)->vbox, &box_req); requisition->height += box_req.height + 3; gtk_gaim_status_box_size_allocate(GtkWidget *widget, GtkAllocation *allocation) GtkRequisition req = {0,0}; GtkAllocation parent_alc, box_alc; combo_box_size_request(widget, &req); box_alc.height = MAX(1, (allocation->height - req.height - 6)); box_alc.y += req.height + 6; gtk_widget_size_allocate((GTK_GAIM_STATUS_BOX(widget))->vbox, &box_alc); parent_alc = *allocation; parent_alc.height = MAX(1,req.height); combo_box_size_allocate(widget, &parent_alc); gtk_widget_size_allocate((GTK_GAIM_STATUS_BOX(widget))->toggle_button, &parent_alc); widget->allocation = *allocation; gtk_gaim_status_box_expose_event(GtkWidget *widget, GtkGaimStatusBox *status_box = GTK_GAIM_STATUS_BOX(widget); gtk_container_propagate_expose(GTK_CONTAINER(widget), status_box->vbox, event); gtk_container_propagate_expose(GTK_CONTAINER(widget), status_box->toggle_button, event); gtk_gaim_status_box_forall(GtkContainer *container, gboolean include_internals, GtkGaimStatusBox *status_box = GTK_GAIM_STATUS_BOX (container); (* callback) (status_box->vbox, callback_data); (* callback) (status_box->toggle_button, callback_data); (* callback) (status_box->arrow, callback_data); combo_box_forall(container, include_internals, callback, callback_data); gtk_gaim_status_box_new() return g_object_new(GTK_GAIM_TYPE_STATUS_BOX, NULL); gtk_gaim_status_box_new_with_account(GaimAccount *account) return g_object_new(GTK_GAIM_TYPE_STATUS_BOX, "account", account, NULL); * Add a row to the dropdown menu. * @param status_box The status box itself. * @param type A GtkGaimStatusBoxItemType. * @param pixbuf The icon to associate with this row in the menu. * @param title The title of this item. For the primitive entries, * this is something like "Available" or "Away." For * the saved statuses, this is something like * "My favorite away message!" This should be * plaintext (non-markedup) (this function escapes it). * @param desc The secondary text for this item. This will be * placed on the row below the title, in a dimmer * font (generally gray). This text should be plaintext * (non-markedup) (this function escapes it). * @param data Data to be associated with this row in the dropdown * menu. For primitives this is the value of the * GaimStatusPrimitive. For saved statuses this is the gtk_gaim_status_box_add(GtkGaimStatusBox *status_box, GtkGaimStatusBoxItemType type, GdkPixbuf *pixbuf, const char *title, const char *desc, gpointer data) text = g_markup_escape_text(title, -1); gboolean show_buddy_icons; gchar *escaped_title, *escaped_desc; show_buddy_icons = gaim_prefs_get_bool("/gaim/gtk/blist/show_buddy_icons"); style = gtk_widget_get_style(GTK_WIDGET(status_box)); snprintf(aa_color, sizeof(aa_color), "#%02x%02x%02x", style->text_aa[GTK_STATE_NORMAL].red >> 8, style->text_aa[GTK_STATE_NORMAL].green >> 8, style->text_aa[GTK_STATE_NORMAL].blue >> 8); escaped_title = g_markup_escape_text(title, -1); escaped_desc = g_markup_escape_text(desc, -1); text = g_strdup_printf("%s%s<span color=\"%s\" size=\"smaller\">%s</span>", show_buddy_icons ? "\n" : " - ", gtk_list_store_append(status_box->dropdown_store, &iter); gtk_list_store_set(status_box->dropdown_store, &iter, gtk_gaim_status_box_add_separator(GtkGaimStatusBox *status_box) /* Don't do anything unless GTK actually supports * gtk_combo_box_set_row_separator_func */ #if GTK_CHECK_VERSION(2,6,0) gtk_list_store_append(status_box->dropdown_store, &iter); gtk_list_store_set(status_box->dropdown_store, &iter, TYPE_COLUMN, GTK_GAIM_STATUS_BOX_TYPE_SEPARATOR, gtk_gaim_status_box_set_connecting(GtkGaimStatusBox *status_box, gboolean connecting) status_box->connecting = connecting; gtk_gaim_status_box_refresh(status_box); gtk_gaim_status_box_pulse_connecting(GtkGaimStatusBox *status_box) if (status_box->connecting_index == 3) status_box->connecting_index = 0; status_box->connecting_index++; gtk_gaim_status_box_refresh(status_box); gtk_gaim_status_box_pulse_typing(GtkGaimStatusBox *status_box) if (status_box->typing_index == 3) status_box->typing_index = 0; status_box->typing_index++; gtk_gaim_status_box_refresh(status_box); find_status_type_by_index(const GaimAccount *account, gint active) const GList *l = gaim_account_get_status_types(account); for (i = 0; l; l = l->next) { GaimStatusType *status_type = l->data; if (!gaim_status_type_is_user_settable(status_type)) message_changed(const char *one, const char *two) if (one == NULL && two == NULL) if (one == NULL || two == NULL) return (g_utf8_collate(one, two) != 0); activate_currently_selected_status(GtkGaimStatusBox *status_box) GtkGaimStatusBoxItemType type; GaimSavedStatus *saved_status; if (!gtk_combo_box_get_active_iter(GTK_COMBO_BOX(status_box), &iter)) gtk_tree_model_get(GTK_TREE_MODEL(status_box->dropdown_store), &iter, * If the currently selected status is "New..." or * "Saved..." or a popular status then do nothing. * activated elsewhere, and we update the status_box * accordingly by monitoring the preference * "/core/savedstatus/current" and then calling * status_menu_refresh_iter() if (type != GTK_GAIM_STATUS_BOX_TYPE_PRIMITIVE) gtk_tree_model_get(GTK_TREE_MODEL(status_box->dropdown_store), &iter, TITLE_COLUMN, &title, -1); message = gtk_gaim_status_box_get_message(status_box); if (!message || !*message) gtk_widget_hide_all(status_box->vbox); status_box->imhtml_visible = FALSE; if (status_box->account == NULL) { /* Save the newly selected status to prefs.xml and status.xml */ /* Has the status really been changed? */ saved_status = gaim_savedstatus_get_current(); if (gaim_savedstatus_get_type(saved_status) == GPOINTER_TO_INT(data)) if (!message_changed(gaim_savedstatus_get_message(saved_status), message)) /* If we've used this type+message before, lookup the transient status */ saved_status = gaim_savedstatus_find_transient_by_type_and_message( GPOINTER_TO_INT(data), message); /* If this type+message is unique then create a new transient saved status */ if (saved_status == NULL) saved_status = gaim_savedstatus_new(NULL, GPOINTER_TO_INT(data)); gaim_savedstatus_set_message(saved_status, message); /* Set the status for each account */ gaim_savedstatus_activate(saved_status); GaimStatusType *status_type; status = gaim_account_get_active_status(status_box->account); g_object_get(G_OBJECT(status_box), "active", &active, NULL); status_type = find_status_type_by_index(status_box->account, active); id = gaim_status_type_get_id(status_type); if (strncmp(id, gaim_status_get_id(status), strlen(id)) == 0) /* Selected status and previous status is the same */ if (!message_changed(message, gaim_status_get_attr_string(status, "message"))) gaim_account_set_status(status_box->account, id, TRUE, "message", message, NULL); gaim_account_set_status(status_box->account, id, static void update_size(GtkGaimStatusBox *status_box) int pad_top, pad_inside, pad_bottom; if (!status_box->imhtml_visible) if (status_box->vbox != NULL) gtk_widget_set_size_request(status_box->vbox, -1, -1); buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(status_box->imhtml)); gtk_text_buffer_get_start_iter(buffer, &iter); while (gtk_text_view_forward_display_line(GTK_TEXT_VIEW(status_box->imhtml), &iter)) lines = gtk_text_buffer_get_line_count(buffer); /* Show a maximum of 4 lines */ wrapped_lines = MIN(wrapped_lines, 4); gtk_text_buffer_get_start_iter(buffer, &iter); gtk_text_view_get_iter_location(GTK_TEXT_VIEW(status_box->imhtml), &iter, &oneline); pad_top = gtk_text_view_get_pixels_above_lines(GTK_TEXT_VIEW(status_box->imhtml)); pad_bottom = gtk_text_view_get_pixels_below_lines(GTK_TEXT_VIEW(status_box->imhtml)); pad_inside = gtk_text_view_get_pixels_inside_wrap(GTK_TEXT_VIEW(status_box->imhtml)); height = (oneline.height + pad_top + pad_bottom) * lines; height += (oneline.height + pad_inside) * (wrapped_lines - lines); gtk_widget_set_size_request(status_box->vbox, -1, height + GAIM_HIG_BOX_SPACE); static void remove_typing_cb(GtkGaimStatusBox *status_box) if (status_box->typing == 0) /* Nothing has changed, so we don't need to do anything */ status_menu_refresh_iter(status_box); activate_currently_selected_status(status_box); g_source_remove(status_box->typing); gtk_gaim_status_box_refresh(status_box); static void gtk_gaim_status_box_changed(GtkComboBox *box) GtkGaimStatusBox *status_box; GtkGaimStatusBoxItemType type; GList *accounts = NULL, *node; status_box = GTK_GAIM_STATUS_BOX(box); if (!gtk_combo_box_get_active_iter(GTK_COMBO_BOX(status_box), &iter)) gtk_tree_model_get(GTK_TREE_MODEL(status_box->dropdown_store), &iter, if (status_box->typing != 0) g_source_remove(status_box->typing); if (GTK_WIDGET_IS_SENSITIVE(GTK_WIDGET(status_box))) if (type == GTK_GAIM_STATUS_BOX_TYPE_POPULAR) saved = gaim_savedstatus_find_by_creation_time(GPOINTER_TO_INT(data)); g_return_if_fail(saved != NULL); gaim_savedstatus_activate(saved); if (type == GTK_GAIM_STATUS_BOX_TYPE_CUSTOM) GaimSavedStatus *saved_status; saved_status = gaim_savedstatus_get_current(); gaim_gtk_status_editor_show(FALSE, gaim_savedstatus_is_transient(saved_status) status_menu_refresh_iter(status_box); if (type == GTK_GAIM_STATUS_BOX_TYPE_SAVED) gaim_gtk_status_window_show(); status_menu_refresh_iter(status_box); * Show the message box whenever the primitive allows for a * message attribute on any protocol that is enabled, * or our protocol, if we have account set accounts = g_list_prepend(accounts, status_box->account); accounts = gaim_accounts_get_all_active(); status_box->imhtml_visible = FALSE; for (node = accounts; node != NULL; node = node->next) GaimStatusType *status_type; status_type = gaim_account_get_status_type_with_primitive(account, GPOINTER_TO_INT(data)); if ((status_type != NULL) && (gaim_status_type_get_attr(status_type, "message") != NULL)) status_box->imhtml_visible = TRUE; if (GTK_WIDGET_IS_SENSITIVE(GTK_WIDGET(status_box))) if (status_box->imhtml_visible) gtk_widget_show_all(status_box->vbox); if (GTK_WIDGET_IS_SENSITIVE(GTK_WIDGET(status_box))) { status_box->typing = g_timeout_add(TYPING_TIMEOUT, (GSourceFunc)remove_typing_cb, status_box); gtk_widget_grab_focus(status_box->imhtml); buf = gtk_text_view_get_buffer(GTK_TEXT_VIEW(status_box->imhtml)); gtk_text_buffer_get_start_iter(buf, &start); gtk_text_buffer_get_end_iter(buf, &end); gtk_text_buffer_move_mark_by_name(buf, "insert", &end); gtk_text_buffer_move_mark_by_name(buf, "selection_bound", &start); gtk_widget_hide_all(status_box->vbox); if (GTK_WIDGET_IS_SENSITIVE(GTK_WIDGET(status_box))) activate_currently_selected_status(status_box); /* This is where we actually set the status */ gtk_gaim_status_box_refresh(status_box); get_statusbox_index(GtkGaimStatusBox *box, GaimSavedStatus *saved_status) switch (gaim_savedstatus_get_type(saved_status)) case GAIM_STATUS_AVAILABLE: case GAIM_STATUS_INVISIBLE: case GAIM_STATUS_OFFLINE: static void imhtml_changed_cb(GtkTextBuffer *buffer, void *data) GtkGaimStatusBox *status_box = (GtkGaimStatusBox*)data; if (GTK_WIDGET_IS_SENSITIVE(GTK_WIDGET(status_box))) if (status_box->typing != 0) { gtk_gaim_status_box_pulse_typing(status_box); g_source_remove(status_box->typing); status_box->typing = g_timeout_add(TYPING_TIMEOUT, (GSourceFunc)remove_typing_cb, status_box); gtk_gaim_status_box_refresh(status_box); static void imhtml_format_changed_cb(GtkIMHtml *imhtml, GtkIMHtmlButtons buttons, void *data) imhtml_changed_cb(NULL, data); char *gtk_gaim_status_box_get_message(GtkGaimStatusBox *status_box) if (status_box->imhtml_visible) return gtk_imhtml_get_markup(GTK_IMHTML(status_box->imhtml));