diff options
author | Gary Kramlich <grim@reaperworld.com> | 2022-08-18 23:27:55 -0500 |
---|---|---|
committer | Gary Kramlich <grim@reaperworld.com> | 2022-08-18 23:27:55 -0500 |
commit | c8ca565f17e82bedbf68599619b0e0230cb6031a (patch) | |
tree | e11d6005f46d155986a2b45af352c1f20da68767 /pidgin/pidginaddchatdialog.c | |
parent | 5f974c106707921e9f9c7e0fabd9c421ab4ae106 (diff) | |
download | pidgin-c8ca565f17e82bedbf68599619b0e0230cb6031a.tar.gz |
Create a new PidginAddChatDialog widget.
This is a full replacement of the existing code in gtkblist.c. Some of the code
remains as it was reused by the join chat dialog code.
Testing Done:
* Added/Removed a chat on XMPP under valgrind
* Opened the roomlist from the dialog with both IRC and XMPP accounts selected.
Reviewed at https://reviews.imfreedom.org/r/1590/
Diffstat (limited to 'pidgin/pidginaddchatdialog.c')
-rw-r--r-- | pidgin/pidginaddchatdialog.c | 464 |
1 files changed, 464 insertions, 0 deletions
diff --git a/pidgin/pidginaddchatdialog.c b/pidgin/pidginaddchatdialog.c new file mode 100644 index 0000000000..3dd5d2294f --- /dev/null +++ b/pidgin/pidginaddchatdialog.c @@ -0,0 +1,464 @@ +/* + * Pidgin - Internet Messenger + * Copyright (C) Pidgin Developers <devel@pidgin.im> + * + * Pidgin is the legal property of its developers, whose names are too numerous + * to list here. Please refer to the COPYRIGHT file distributed with this + * source distribution. + * + * 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, see <https://www.gnu.org/licenses/>. + */ + +#include <glib/gi18n.h> + +#include "pidginaddchatdialog.h" + +#include "gtkaccount.h" +#include "gtkroomlist.h" +#include "pidginaccountchooser.h" +#include "pidginaccountstore.h" +#include "pidgincore.h" + +struct _PidginAddChatDialog { + GtkDialog parent; + + GtkTreeModel *filter; + + const gchar *default_name; + + GtkWidget *account; + GtkWidget *dynamic_box; + GtkWidget *alias; + GtkWidget *group; + GtkWidget *autojoin; + GtkWidget *persistent; + GtkSizeGroup *sg; + + GList *inputs; +}; + +G_DEFINE_TYPE(PidginAddChatDialog, pidgin_add_chat_dialog, GTK_TYPE_DIALOG) + +/* ugh, prototypes... */ +static void pidgin_add_chat_dialog_remove_widget_cb(GtkWidget *child, gpointer data); +static void pidgin_add_chat_dialog_input_changed_cb(GtkEditable *editable, gpointer data); + +/****************************************************************************** + * Helpers + *****************************************************************************/ +static void +pidgin_add_chat_dialog_validate_input(gpointer data, gpointer user_data) { + gboolean *valid = user_data; + + if(!g_object_get_data(data, "integer")) { + gboolean required = FALSE; + + required = GPOINTER_TO_INT(g_object_get_data(data, "required")); + if(required) { + const gchar *value = NULL; + + value = gtk_entry_get_text(GTK_ENTRY(data)); + if(value == NULL || *value == '\0') { + *valid = FALSE; + } + } + } +} + +static void +pidgin_add_chat_dialog_validate(PidginAddChatDialog *dialog) { + gboolean valid = TRUE; + + /* The callback should only set valid to FALSE if a field is invalid and + * not set valid if the field is valid. + */ + g_list_foreach(dialog->inputs, + pidgin_add_chat_dialog_validate_input, + &valid); + + gtk_dialog_set_response_sensitive(GTK_DIALOG(dialog), GTK_RESPONSE_OK, + valid); +} + +static void +pidgin_add_chat_dialog_update_components(PidginAddChatDialog *dialog) { + PurpleAccount *account = NULL; + PurpleConnection *connection = NULL; + PurpleProtocol *protocol = NULL; + GList *info = NULL; + GHashTable *defaults = NULL; + gboolean focus_set = FALSE; + + account = pidgin_account_chooser_get_selected(PIDGIN_ACCOUNT_CHOOSER(dialog->account)); + connection = purple_account_get_connection(account); + protocol = purple_account_get_protocol(account); + + /* Clean up the dynamic box and our list of entires. */ + gtk_container_foreach(GTK_CONTAINER(dialog->dynamic_box), + pidgin_add_chat_dialog_remove_widget_cb, + dialog->dynamic_box); + g_clear_pointer(&dialog->inputs, g_list_free); + + info = purple_protocol_chat_info(PURPLE_PROTOCOL_CHAT(protocol), + connection); + defaults = purple_protocol_chat_info_defaults(PURPLE_PROTOCOL_CHAT(protocol), + connection, + dialog->default_name); + + while(info != NULL) { + GtkWidget *box = NULL, *label = NULL, *input = NULL; + PurpleProtocolChatEntry *pce = info->data; + + if(pce->is_int) { + GtkAdjustment *adjustment = gtk_adjustment_new(pce->min, pce->min, + pce->max, 1, 10, 10); + input = gtk_spin_button_new(adjustment, 1, 0); + } else { + gchar *value = NULL; + + input = gtk_entry_new(); + g_signal_connect(input, "changed", + G_CALLBACK(pidgin_add_chat_dialog_input_changed_cb), + dialog); + + value = g_hash_table_lookup(defaults, pce->identifier); + if(value != NULL) { + gtk_entry_set_text(GTK_ENTRY(input), value); + } + + if(pce->secret) { + gtk_entry_set_visibility(GTK_ENTRY(input), FALSE); + gtk_entry_set_input_purpose(GTK_ENTRY(input), + GTK_INPUT_PURPOSE_PASSWORD); + } + } + + box = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 6); + gtk_box_pack_start(GTK_BOX(dialog->dynamic_box), box, FALSE, FALSE, 0); + + label = gtk_label_new_with_mnemonic(pce->label); + gtk_label_set_xalign(GTK_LABEL(label), 0.0f); + gtk_label_set_yalign(GTK_LABEL(label), 0.0f); + gtk_box_pack_start(GTK_BOX(box), label, FALSE, TRUE, 0); + gtk_size_group_add_widget(dialog->sg, label); + + gtk_box_pack_start(GTK_BOX(box), input, TRUE, TRUE, 0); + if(!focus_set) { + gtk_widget_grab_focus(input); + focus_set = TRUE; + } + + g_object_set_data(G_OBJECT(input), "identifier", + (gpointer)pce->identifier); + g_object_set_data(G_OBJECT(input), "integer", + GINT_TO_POINTER(pce->is_int)); + g_object_set_data(G_OBJECT(input), "required", + GINT_TO_POINTER(pce->required)); + + dialog->inputs = g_list_append(dialog->inputs, input); + + g_free(pce); + + info = g_list_delete_link(info, info); + } + + g_hash_table_destroy(defaults); + + gtk_widget_show_all(dialog->dynamic_box); + + pidgin_add_chat_dialog_validate(dialog); +} + +static gboolean +pidgin_add_chat_dialog_filter_accounts(GtkTreeModel *model, GtkTreeIter *iter, + gpointer data) +{ + PurpleAccount *account = NULL; + PurpleProtocol *protocol = NULL; + gboolean ret = FALSE; + + g_return_val_if_fail(GTK_IS_TREE_MODEL(model), FALSE); + g_return_val_if_fail(iter != NULL, FALSE); + + gtk_tree_model_get(model, iter, PIDGIN_ACCOUNT_STORE_COLUMN_ACCOUNT, + &account, -1); + + if(!PURPLE_IS_ACCOUNT(account)) { + return FALSE; + } + + protocol = purple_account_get_protocol(account); + if(PURPLE_IS_PROTOCOL(protocol)) { + ret = PURPLE_PROTOCOL_IMPLEMENTS(protocol, CHAT, info); + } + + g_object_unref(G_OBJECT(account)); + + return ret; +} + +static void +pidgin_add_chat_dialog_add_input_to_components(gpointer data, + gpointer user_data) +{ + GHashTable *components = user_data; + gchar *identifier = NULL; + gchar *value = NULL; + + identifier = g_strdup(g_object_get_data(data, "identifier")); + + if(g_object_get_data(data, "integer")) { + gint int_value = gtk_spin_button_get_value_as_int(GTK_SPIN_BUTTON(data)); + + value = g_strdup_printf("%d", int_value); + } else { + const gchar *str_value = gtk_entry_get_text(GTK_ENTRY(data)); + + if(*str_value != '\0') { + value = g_strdup(str_value); + } + } + + /* If the value was changed to an empty string, we should remove it from the + * components. + */ + if(value == NULL) { + g_hash_table_remove(components, identifier); + g_free(identifier); + } else { + g_hash_table_replace(components, identifier, value); + } +} + +/****************************************************************************** + * Callbacks + *****************************************************************************/ +static void +pidgin_add_chat_dialog_input_changed_cb(G_GNUC_UNUSED GtkEditable *editable, + gpointer data) +{ + pidgin_add_chat_dialog_validate(data); +} + +static void +pidgin_add_chat_dialog_remove_widget_cb(GtkWidget *child, gpointer data) { + GtkContainer *parent = data; + + gtk_container_remove(parent, child); +} + +static void +pidgin_add_chat_dialog_response_cb(GtkDialog *dialog, gint response_id, + G_GNUC_UNUSED gpointer data) +{ + PurpleAccount *account = NULL; + PidginAddChatDialog *acdialog = PIDGIN_ADD_CHAT_DIALOG(dialog); + gboolean close = TRUE; + + account = pidgin_account_chooser_get_selected(PIDGIN_ACCOUNT_CHOOSER(acdialog->account)); + + if(response_id == 1) { + pidgin_roomlist_dialog_show_with_account(account); + + close = FALSE; + } else if(response_id == GTK_RESPONSE_OK) { + PurpleChat *chat = NULL; + GHashTable *components = NULL; + const gchar *alias = NULL; + + components = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, + g_free); + g_list_foreach(acdialog->inputs, + pidgin_add_chat_dialog_add_input_to_components, + components); + + alias = gtk_entry_get_text(GTK_ENTRY(acdialog->alias)); + + chat = purple_chat_new(account, alias, components); + + if(PURPLE_IS_CHAT(chat)) { + PurpleGroup *group = NULL; + gchar *group_name = NULL; + + group_name = gtk_combo_box_text_get_active_text(GTK_COMBO_BOX_TEXT(acdialog->group)); + if(group_name != NULL && *group_name != '\0') { + group = purple_blist_find_group(group_name); + } + + if(!PURPLE_IS_GROUP(group)) { + group = purple_group_new(group_name); + purple_blist_add_group(group, NULL); + } + + purple_blist_add_chat(chat, group, NULL); + + if(gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(acdialog->autojoin))) { + purple_blist_node_set_bool(PURPLE_BLIST_NODE(chat), + "gtk-autojoin", TRUE); + } + if(gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(acdialog->persistent))) { + purple_blist_node_set_bool(PURPLE_BLIST_NODE(chat), + "gtk-persistent", TRUE); + } + + g_free(group_name); + } else { + g_warning("failed to create chat"); + } + } + + if(close) { + gtk_widget_destroy(GTK_WIDGET(dialog)); + } +} + +static void +pidgin_add_chat_dialog_account_changed_cb(GtkComboBox *widget, gpointer data) { + PidginAddChatDialog *dialog = data; + PurpleAccount *account = NULL; + + account = pidgin_account_chooser_get_selected(PIDGIN_ACCOUNT_CHOOSER(widget)); + + if(PURPLE_IS_ACCOUNT(account)) { + PurpleProtocol *protocol = purple_account_get_protocol(account); + + if(PURPLE_IS_PROTOCOL(protocol)) { + gboolean roomlist = FALSE; + + roomlist = PURPLE_PROTOCOL_IMPLEMENTS(protocol, ROOMLIST, get_list); + + gtk_dialog_set_response_sensitive(GTK_DIALOG(dialog), 1, roomlist); + } + } + + pidgin_add_chat_dialog_update_components(dialog); + + pidgin_add_chat_dialog_validate(dialog); +} + +static void +pidgin_add_chat_dialog_username_changed_cb(G_GNUC_UNUSED GtkEditable *editable, + gpointer data) +{ + pidgin_add_chat_dialog_validate(data); +} + +static void +pidgin_add_chat_dialog_group_cb(PurpleBlistNode *node, gpointer data) { + PidginAddChatDialog *dialog = data; + PurpleGroup *group = PURPLE_GROUP(node); + + gtk_combo_box_text_append_text(GTK_COMBO_BOX_TEXT(dialog->group), + purple_group_get_name(group)); +} + +/****************************************************************************** + * GObject Implementation + *****************************************************************************/ +static void +pidgin_add_chat_dialog_finalize(GObject *obj) { + PidginAddChatDialog *dialog = PIDGIN_ADD_CHAT_DIALOG(obj); + + g_list_free(dialog->inputs); + + G_OBJECT_CLASS(pidgin_add_chat_dialog_parent_class)->finalize(obj); +} + +static void +pidgin_add_chat_dialog_init(PidginAddChatDialog *dialog) { + gtk_widget_init_template(GTK_WIDGET(dialog)); + + gtk_dialog_set_default_response(GTK_DIALOG(dialog), GTK_RESPONSE_OK); + + gtk_tree_model_filter_set_visible_func(GTK_TREE_MODEL_FILTER(dialog->filter), + pidgin_add_chat_dialog_filter_accounts, + NULL, NULL); + gtk_tree_model_filter_refilter(GTK_TREE_MODEL_FILTER(dialog->filter)); + + gtk_combo_box_set_active(GTK_COMBO_BOX(dialog->account), 0); + + purple_blist_walk(pidgin_add_chat_dialog_group_cb, NULL, NULL, NULL, + dialog); +} + +static void +pidgin_add_chat_dialog_class_init(PidginAddChatDialogClass *klass) { + GObjectClass *obj_class = G_OBJECT_CLASS(klass); + GtkWidgetClass *widget_class = GTK_WIDGET_CLASS(klass); + + obj_class->finalize = pidgin_add_chat_dialog_finalize; + + gtk_widget_class_set_template_from_resource( + widget_class, + "/im/pidgin/Pidgin3/Dialogs/addchat.ui" + ); + + gtk_widget_class_bind_template_child(widget_class, PidginAddChatDialog, + filter); + + gtk_widget_class_bind_template_child(widget_class, PidginAddChatDialog, + account); + gtk_widget_class_bind_template_child(widget_class, PidginAddChatDialog, + dynamic_box); + gtk_widget_class_bind_template_child(widget_class, PidginAddChatDialog, + alias); + gtk_widget_class_bind_template_child(widget_class, PidginAddChatDialog, + group); + gtk_widget_class_bind_template_child(widget_class, PidginAddChatDialog, + autojoin); + gtk_widget_class_bind_template_child(widget_class, PidginAddChatDialog, + persistent); + gtk_widget_class_bind_template_child(widget_class, PidginAddChatDialog, + sg); + + gtk_widget_class_bind_template_callback(widget_class, + pidgin_add_chat_dialog_response_cb); + gtk_widget_class_bind_template_callback(widget_class, + pidgin_add_chat_dialog_account_changed_cb); + gtk_widget_class_bind_template_callback(widget_class, + pidgin_add_chat_dialog_username_changed_cb); +} + +/****************************************************************************** + * Public API + *****************************************************************************/ +GtkWidget * +pidgin_add_chat_dialog_new(PurpleAccount *account, PurpleGroup *group, + const gchar *alias, const gchar *name) +{ + GtkWidget *dialog = g_object_new(PIDGIN_TYPE_ADD_CHAT_DIALOG, NULL); + PidginAddChatDialog *acdialog = PIDGIN_ADD_CHAT_DIALOG(dialog); + + if(PURPLE_IS_ACCOUNT(account)) { + pidgin_account_chooser_set_selected(PIDGIN_ACCOUNT_CHOOSER(acdialog->account), + account); + } + + if(alias != NULL) { + gtk_entry_set_text(GTK_ENTRY(acdialog->alias), alias); + } + + if(PURPLE_IS_GROUP(group)) { + GtkWidget *entry = NULL; + + entry = gtk_bin_get_child(GTK_BIN(acdialog->group)); + gtk_entry_set_text(GTK_ENTRY(entry), purple_group_get_name(group)); + } + + acdialog->default_name = name; + + pidgin_add_chat_dialog_update_components(acdialog); + + return dialog; +} |