Mercurial > pidgin
view console/gntstatus.c @ 14278:f71894e1700d
[gaim-migrate @ 16963]
Fix a crazy bug that we found at meebo. I'm not sure it
affects gtk Gaim in quite the same way, but what was
happeing is that the conversation sequence number should
be incremented by 1 each time a new conversation is
created (whether created by someone else or created by
you). However, in one place in Gaim conv_seq was being
incremented BEFORE it was used and in another it was
being incremented AFTER it was used. This can lead to
a sequence number being used twice.
committer: Tailor Script <tailor@pidgin.im>
author | Mark Doliner <mark@kingant.net> |
---|---|
date | Mon, 21 Aug 2006 22:50:58 +0000 |
parents | 789d80a6b9d9 |
children | 81650a27f253 |
line wrap: on
line source
#include <gnt.h> #include <gntbox.h> #include <gntbutton.h> #include <gntcombobox.h> #include <gntentry.h> #include <gntlabel.h> #include <gntline.h> #include <gnttree.h> #include <notify.h> #include <request.h> #include "gntgaim.h" #include "gntstatus.h" static struct { GntWidget *window; GntWidget *tree; } statuses; typedef struct { GaimSavedStatus *saved; GntWidget *window; GntWidget *title; GntWidget *type; GntWidget *message; GntWidget *tree; GHashTable *hash; /* list of windows for substatuses */ } EditStatus; typedef struct { GntWidget *window; GntWidget *type; GntWidget *message; EditStatus *parent; GaimAccount *account; } EditSubStatus; static GList *edits; /* List of opened edit-status dialogs */ static void reset_status_window(GntWidget *widget, gpointer null) { statuses.window = NULL; statuses.tree = NULL; } static void populate_statuses(GntTree *tree) { const GList *list; for (list = gaim_savedstatuses_get_all(); list; list = list->next) { GaimSavedStatus *saved = list->data; const char *title, *type, *message; if (gaim_savedstatus_is_transient(saved)) continue; title = gaim_savedstatus_get_title(saved); type = gaim_primitive_get_name_from_type(gaim_savedstatus_get_type(saved)); message = gaim_savedstatus_get_message(saved); /* XXX: Strip possible markups */ gnt_tree_add_row_last(tree, saved, gnt_tree_create_row(tree, title, type, message), NULL); } } static void really_delete_status(GaimSavedStatus *saved) { GList *iter; for (iter = edits; iter; iter = iter->next) { EditStatus *edit = iter->data; if (edit->saved == saved) { gnt_widget_destroy(edit->window); break; } } if (statuses.tree) gnt_tree_remove(GNT_TREE(statuses.tree), saved); gaim_savedstatus_delete(gaim_savedstatus_get_title(saved)); } static void ask_before_delete(GntWidget *button, gpointer null) { char *ask; GaimSavedStatus *saved; g_return_if_fail(statuses.tree != NULL); saved = gnt_tree_get_selection_data(GNT_TREE(statuses.tree)); ask = g_strdup_printf(_("Are you sure you want to delete \"%s\""), gaim_savedstatus_get_title(saved)); gaim_request_action(saved, _("Delete Status"), ask, NULL, 0, saved, 2, _("Delete"), really_delete_status, _("Cancel"), NULL); g_free(ask); } static void use_savedstatus_cb(GntWidget *widget, gpointer null) { g_return_if_fail(statuses.tree != NULL); gaim_savedstatus_activate(gnt_tree_get_selection_data(GNT_TREE(statuses.tree))); } static void edit_savedstatus_cb(GntWidget *widget, gpointer null) { g_return_if_fail(statuses.tree != NULL); gg_savedstatus_edit(gnt_tree_get_selection_data(GNT_TREE(statuses.tree))); } void gg_savedstatus_show_all() { GntWidget *window, *tree, *box, *button; if (statuses.window) return; statuses.window = window = gnt_vbox_new(FALSE); gnt_box_set_toplevel(GNT_BOX(window), TRUE); gnt_box_set_title(GNT_BOX(window), _("Saved Statuses")); gnt_box_set_fill(GNT_BOX(window), FALSE); gnt_box_set_alignment(GNT_BOX(window), GNT_ALIGN_MID); gnt_box_set_pad(GNT_BOX(window), 0); /* XXX: Add some sorting function to sort alphabetically, perhaps */ statuses.tree = tree = gnt_tree_new_with_columns(3); gnt_tree_set_column_titles(GNT_TREE(tree), _("Title"), _("Type"), _("Message")); gnt_tree_set_show_title(GNT_TREE(tree), TRUE); gnt_tree_set_col_width(GNT_TREE(tree), 0, 25); gnt_tree_set_col_width(GNT_TREE(tree), 1, 12); gnt_tree_set_col_width(GNT_TREE(tree), 2, 35); gnt_box_add_widget(GNT_BOX(window), tree); populate_statuses(GNT_TREE(tree)); box = gnt_hbox_new(FALSE); gnt_box_add_widget(GNT_BOX(window), box); button = gnt_button_new(_("Use")); gnt_box_add_widget(GNT_BOX(box), button); g_signal_connect(G_OBJECT(button), "activate", G_CALLBACK(use_savedstatus_cb), NULL); button = gnt_button_new(_("Add")); gnt_box_add_widget(GNT_BOX(box), button); g_signal_connect_swapped(G_OBJECT(button), "activate", G_CALLBACK(gg_savedstatus_edit), NULL); button = gnt_button_new(_("Edit")); gnt_box_add_widget(GNT_BOX(box), button); g_signal_connect(G_OBJECT(button), "activate", G_CALLBACK(edit_savedstatus_cb), NULL); button = gnt_button_new(_("Delete")); gnt_box_add_widget(GNT_BOX(box), button); g_signal_connect(G_OBJECT(button), "activate", G_CALLBACK(ask_before_delete), NULL); button = gnt_button_new(_("Close")); gnt_box_add_widget(GNT_BOX(box), button); g_signal_connect_swapped(G_OBJECT(button), "activate", G_CALLBACK(gnt_widget_destroy), window); g_signal_connect(G_OBJECT(window), "destroy", G_CALLBACK(reset_status_window), NULL); gnt_widget_show(window); } static void destroy_substatus_win(GaimAccount *account, EditSubStatus *sub, gpointer null) { gnt_widget_destroy(sub->window); /* the "destroy" callback will remove entry from the hashtable */ } static void update_edit_list(GntWidget *widget, EditStatus *edit) { edits = g_list_remove(edits, edit); gaim_notify_close_with_handle(edit); g_hash_table_foreach(edit->hash, (GHFunc)destroy_substatus_win, NULL); g_free(edit); } static void save_savedstatus_cb(GntWidget *button, EditStatus *edit) { const char *title, *message; GaimStatusPrimitive prim; GaimSavedStatus *find; title = gnt_entry_get_text(GNT_ENTRY(edit->title)); message = gnt_entry_get_text(GNT_ENTRY(edit->message)); if (!message || !*message) message = NULL; prim = GPOINTER_TO_INT(gnt_combo_box_get_selected_data(GNT_COMBO_BOX(edit->type))); if (!title || !*title) { gaim_notify_error(edit, _("Error"), _("Invalid title"), _("Please enter a non-empty title for the status.")); return; } find = gaim_savedstatus_find(title); if (find && find != edit->saved) { gaim_notify_error(edit, _("Error"), _("Duplicate title"), _("Please enter a different title for the status.")); return; } if (edit->saved == NULL) { edit->saved = gaim_savedstatus_new(title, prim); gaim_savedstatus_set_message(edit->saved, message); if (statuses.tree) gnt_tree_add_row_last(GNT_TREE(statuses.tree), edit->saved, gnt_tree_create_row(GNT_TREE(statuses.tree), title, gaim_primitive_get_name_from_type(prim), message), NULL); } else { gaim_savedstatus_set_title(edit->saved, title); gaim_savedstatus_set_type(edit->saved, prim); gaim_savedstatus_set_message(edit->saved, message); if (statuses.tree) { gnt_tree_change_text(GNT_TREE(statuses.tree), edit->saved, 0, title); gnt_tree_change_text(GNT_TREE(statuses.tree), edit->saved, 1, gaim_primitive_get_name_from_type(prim)); gnt_tree_change_text(GNT_TREE(statuses.tree), edit->saved, 2, message); } } if (g_object_get_data(G_OBJECT(button), "use")) gaim_savedstatus_activate(edit->saved); gnt_widget_destroy(edit->window); } static void add_substatus(EditStatus *edit, GaimAccount *account) { char *name; const char *type = NULL, *message = NULL; GaimSavedStatusSub *sub = NULL; if (!edit || !edit->tree) return; if (edit->saved) sub = gaim_savedstatus_get_substatus(edit->saved, account); if (sub) { type = gaim_status_type_get_name(gaim_savedstatus_substatus_get_type(sub)); message = gaim_savedstatus_substatus_get_message(sub); } name = g_strdup_printf("%s (%s)", gaim_account_get_username(account), gaim_account_get_protocol_name(account)); gnt_tree_add_choice(GNT_TREE(edit->tree), account, gnt_tree_create_row(GNT_TREE(edit->tree), name, type, message), NULL, NULL); if (sub) gnt_tree_set_choice(GNT_TREE(edit->tree), account, TRUE); g_free(name); } static void substatus_window_destroy_cb(GntWidget *window, EditSubStatus *sub) { g_hash_table_remove(sub->parent->hash, sub->account); g_free(sub); } static void save_substatus_cb(GntWidget *widget, EditSubStatus *sub) { GaimSavedStatus *saved = sub->parent->saved; GaimAccount *account = sub->account; const char *message; GaimStatusType *type; type = gnt_combo_box_get_selected_data(GNT_COMBO_BOX(sub->type)); message = gnt_entry_get_text(GNT_ENTRY(sub->message)); gaim_savedstatus_set_substatus(saved, account, type, message); gnt_tree_set_choice(GNT_TREE(sub->parent->tree), account, TRUE); gnt_tree_change_text(GNT_TREE(sub->parent->tree), account, 1, gaim_status_type_get_name(type)); gnt_tree_change_text(GNT_TREE(sub->parent->tree), account, 2, message); gnt_widget_destroy(sub->window); } static gboolean popup_substatus(GntTree *tree, const char *key, EditStatus *edit) { if (key[0] == ' ' && key[1] == 0) { EditSubStatus *sub; GntWidget *window, *combo, *entry, *box, *button, *l; GaimSavedStatusSub *substatus = NULL; const GList *iter; char *name; GaimAccount *account = gnt_tree_get_selection_data(tree); if (gnt_tree_get_choice(tree, account)) { /* There was a savedstatus for this account. Now remove it. */ gaim_savedstatus_unset_substatus(edit->saved, account); gnt_tree_change_text(tree, account, 1, NULL); gnt_tree_change_text(tree, account, 2, NULL); return FALSE; } if (g_hash_table_lookup(edit->hash, account)) return TRUE; if (edit->saved) substatus = gaim_savedstatus_get_substatus(edit->saved, account); sub = g_new0(EditSubStatus, 1); sub->parent = edit; sub->account = account; sub->window = window = gnt_vbox_new(FALSE); gnt_box_set_toplevel(GNT_BOX(window), TRUE); gnt_box_set_title(GNT_BOX(window), _("Substatus")); /* XXX: a better title */ box = gnt_hbox_new(FALSE); gnt_box_add_widget(GNT_BOX(box), gnt_label_new(_("Account:"))); name = g_strdup_printf("%s (%s)", gaim_account_get_username(account), gaim_account_get_protocol_name(account)); gnt_box_add_widget(GNT_BOX(box), gnt_label_new(name)); g_free(name); gnt_box_add_widget(GNT_BOX(window), box); box = gnt_hbox_new(FALSE); gnt_box_add_widget(GNT_BOX(box), (l = gnt_label_new(_("Status:")))); gnt_widget_set_size(l, 0, 1); /* I don't like having to do this */ sub->type = combo = gnt_combo_box_new(); gnt_box_add_widget(GNT_BOX(box), combo); gnt_box_add_widget(GNT_BOX(window), box); for (iter = gaim_account_get_status_types(account); iter; iter = iter->next) { GaimStatusType *type = iter->data; if (!gaim_status_type_is_user_settable(type)) continue; gnt_combo_box_add_data(GNT_COMBO_BOX(combo), type, gaim_status_type_get_name(type)); } box = gnt_hbox_new(FALSE); gnt_box_add_widget(GNT_BOX(box), gnt_label_new(_("Message:"))); sub->message = entry = gnt_entry_new(substatus ? gaim_savedstatus_substatus_get_message(substatus) : NULL); gnt_box_add_widget(GNT_BOX(box), entry); gnt_box_add_widget(GNT_BOX(window), box); box = gnt_hbox_new(FALSE); button = gnt_button_new(_("Cancel")); g_signal_connect_swapped(G_OBJECT(button), "activate", G_CALLBACK(gnt_widget_destroy), window); gnt_box_add_widget(GNT_BOX(box), button); button = gnt_button_new(_("Save")); g_signal_connect(G_OBJECT(button), "activate", G_CALLBACK(save_substatus_cb), sub); gnt_box_add_widget(GNT_BOX(box), button); gnt_box_add_widget(GNT_BOX(window), box); gnt_widget_show(window); g_hash_table_insert(edit->hash, account, sub); g_signal_connect(G_OBJECT(window), "destroy", G_CALLBACK(substatus_window_destroy_cb), sub); return TRUE; } return FALSE; } void gg_savedstatus_edit(GaimSavedStatus *saved) { EditStatus *edit; GntWidget *window, *box, *button, *entry, *combo, *label, *tree; GaimStatusPrimitive prims[] = {GAIM_STATUS_AVAILABLE, GAIM_STATUS_AWAY, GAIM_STATUS_INVISIBLE, GAIM_STATUS_OFFLINE, GAIM_STATUS_UNSET}, current; GList *iter; int i; if (saved) { GList *iter; for (iter = edits; iter; iter = iter->next) { edit = iter->data; if (edit->saved == saved) return; } } edit = g_new0(EditStatus, 1); edit->saved = saved; edit->window = window = gnt_vbox_new(FALSE); gnt_box_set_toplevel(GNT_BOX(window), TRUE); gnt_box_set_title(GNT_BOX(window), _("Edit Status")); gnt_box_set_fill(GNT_BOX(window), TRUE); gnt_box_set_alignment(GNT_BOX(window), GNT_ALIGN_LEFT); gnt_box_set_pad(GNT_BOX(window), 0); edits = g_list_append(edits, edit); /* Title */ box = gnt_hbox_new(FALSE); gnt_box_set_alignment(GNT_BOX(box), GNT_ALIGN_LEFT); gnt_box_add_widget(GNT_BOX(window), box); gnt_box_add_widget(GNT_BOX(box), gnt_label_new(_("Title"))); edit->title = entry = gnt_entry_new(saved ? gaim_savedstatus_get_title(saved) : NULL); gnt_box_add_widget(GNT_BOX(box), entry); /* Type */ box = gnt_hbox_new(FALSE); gnt_box_add_widget(GNT_BOX(window), box); gnt_box_add_widget(GNT_BOX(box), label = gnt_label_new(_("Status"))); gnt_widget_set_size(label, 0, 1); edit->type = combo = gnt_combo_box_new(); gnt_box_add_widget(GNT_BOX(box), combo); current = saved ? gaim_savedstatus_get_type(saved) : GAIM_STATUS_UNSET; for (i = 0; prims[i] != GAIM_STATUS_UNSET; i++) { gnt_combo_box_add_data(GNT_COMBO_BOX(combo), GINT_TO_POINTER(prims[i]), gaim_primitive_get_name_from_type(prims[i])); if (prims[i] == current) gnt_combo_box_set_selected(GNT_COMBO_BOX(combo), GINT_TO_POINTER(current)); } /* Message */ box = gnt_hbox_new(FALSE); gnt_box_add_widget(GNT_BOX(window), box); gnt_box_add_widget(GNT_BOX(box), gnt_label_new(_("Message"))); edit->message = entry = gnt_entry_new(saved ? gaim_savedstatus_get_message(saved) : NULL); gnt_box_add_widget(GNT_BOX(window), entry); gnt_box_add_widget(GNT_BOX(window), gnt_hline_new()); gnt_box_add_widget(GNT_BOX(window), gnt_label_new(_("Use different status for following accounts"))); edit->hash = g_hash_table_new(g_direct_hash, g_direct_equal); edit->tree = tree = gnt_tree_new_with_columns(3); gnt_box_add_widget(GNT_BOX(window), tree); gnt_tree_set_show_title(GNT_TREE(tree), TRUE); gnt_tree_set_column_titles(GNT_TREE(tree), _("Account"), _("Status"), _("Message")); gnt_tree_set_col_width(GNT_TREE(tree), 0, 30); gnt_tree_set_col_width(GNT_TREE(tree), 1, 10); gnt_tree_set_col_width(GNT_TREE(tree), 2, 30); for (iter = gaim_accounts_get_all(); iter; iter = iter->next) { add_substatus(edit, iter->data); } g_signal_connect(G_OBJECT(tree), "key_pressed", G_CALLBACK(popup_substatus), edit); /* The buttons */ box = gnt_hbox_new(FALSE); gnt_box_add_widget(GNT_BOX(window), box); /* Save */ button = gnt_button_new(_("Save")); gnt_box_add_widget(GNT_BOX(box), button); g_object_set_data(G_OBJECT(button), "use", NULL); g_signal_connect(G_OBJECT(button), "activate", G_CALLBACK(save_savedstatus_cb), edit); /* Save & Use */ button = gnt_button_new(_("Save & Use")); gnt_box_add_widget(GNT_BOX(box), button); g_object_set_data(G_OBJECT(button), "use", GINT_TO_POINTER(TRUE)); g_signal_connect(G_OBJECT(button), "activate", G_CALLBACK(save_savedstatus_cb), edit); /* Cancel */ button = gnt_button_new(_("Cancel")); gnt_box_add_widget(GNT_BOX(box), button); g_signal_connect_swapped(G_OBJECT(button), "activate", G_CALLBACK(gnt_widget_destroy), window); g_signal_connect(G_OBJECT(window), "destroy", G_CALLBACK(update_edit_list), edit); gnt_widget_show(window); }