--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/protocols/demo/purpledemoprotocolactions.c Mon Mar 25 21:43:28 2024 -0500 @@ -0,0 +1,950 @@ +/* + * Purple - Internet Messaging Library + * Copyright (C) Pidgin Developers <devel@pidgin.im> + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 2 of the License, or (at your option) any later version. + * + * This library 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 + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, see <https://www.gnu.org/licenses/>. + */ + +#include <glib/gi18n-lib.h> + +#include "purpledemoprotocol.h" +#include "purpledemoprotocolactions.h" +#include "purpledemoresource.h" + +/****************************************************************************** + * Connection failure action implementations + *****************************************************************************/ +#define REAPER_BUDDY_NAME ("Gary") +#define DEFAULT_REAP_TIME (5) /* seconds */ +#define FATAL_TICK_STR N_("Reaping connection in %d second...") +#define FATAL_TICK_PLURAL_STR N_("Reaping connection in %d seconds...") +#define FATAL_DISCONNECT_STR N_("%s reaped the connection") +#define TEMPORARY_TICK_STR N_("Pruning connection in %d second...") +#define TEMPORARY_TICK_PLURAL_STR N_("Pruning connection in %d seconds...") +#define TEMPORARY_DISCONNECT_STR N_("%s pruned the connection") + +static gboolean +purple_demo_protocol_failure_tick(gpointer data, + PurpleConnectionError error_code, + const gchar *tick_str, + const gchar *tick_plural_str, + const gchar *disconnect_str) +{ + PurpleConnection *connection = PURPLE_CONNECTION(data); + PurpleAccount *account = purple_connection_get_account(connection); + gchar *message = NULL; + gint timeout = 0; + + timeout = GPOINTER_TO_INT(g_object_steal_data(G_OBJECT(connection), + "reaping-time")); + timeout--; + if(timeout > 0) { + PurpleContact *contact = NULL; + PurpleContactManager *manager = NULL; + + g_object_set_data(G_OBJECT(connection), "reaping-time", + GINT_TO_POINTER(timeout)); + + manager = purple_contact_manager_get_default(); + contact = purple_contact_manager_find_with_username(manager, account, + REAPER_BUDDY_NAME); + + if(PURPLE_IS_CONTACT(contact)) { + PurpleContactInfo *info = PURPLE_CONTACT_INFO(contact); + PurplePresence *presence = purple_contact_info_get_presence(info); + const char *format = ngettext(tick_str, tick_plural_str, timeout); + + message = g_strdup_printf(format, timeout); + purple_presence_set_message(presence, message); + g_free(message); + } + + return G_SOURCE_CONTINUE; + } + + message = g_strdup_printf(_(disconnect_str), REAPER_BUDDY_NAME); + purple_connection_error(connection, error_code, message); + g_free(message); + + return G_SOURCE_REMOVE; +} + +static gboolean +purple_demo_protocol_fatal_failure_cb(gpointer data) { + return purple_demo_protocol_failure_tick(data, + PURPLE_CONNECTION_ERROR_CUSTOM_FATAL, + FATAL_TICK_STR, + FATAL_TICK_PLURAL_STR, + FATAL_DISCONNECT_STR); +} + +static gboolean +purple_demo_protocol_temporary_failure_cb(gpointer data) { + return purple_demo_protocol_failure_tick(data, + PURPLE_CONNECTION_ERROR_CUSTOM_TEMPORARY, + TEMPORARY_TICK_STR, + TEMPORARY_TICK_PLURAL_STR, + TEMPORARY_DISCONNECT_STR); +} + +static void +purple_demo_protocol_failure_action_activate(G_GNUC_UNUSED GSimpleAction *action, + GVariant *parameter, + const gchar *tick_str, + const gchar *tick_plural_str, + GSourceFunc cb) +{ + PurpleAccountManager *account_manager = NULL; + PurpleAccount *account = NULL; + PurpleConnection *connection = NULL; + PurpleContact *contact = NULL; + PurpleContactManager *contact_manager = NULL; + const char *account_id = NULL; + + if(!g_variant_is_of_type(parameter, G_VARIANT_TYPE_STRING)) { + g_critical("Demo failure action parameter is of incorrect type %s", + g_variant_get_type_string(parameter)); + return; + } + + account_id = g_variant_get_string(parameter, NULL); + account_manager = purple_account_manager_get_default(); + account = purple_account_manager_find_by_id(account_manager, account_id); + connection = purple_account_get_connection(account); + + /* Do nothing if disconnected, or already in process of reaping. */ + if(!PURPLE_IS_CONNECTION(connection)) { + g_clear_object(&account); + + return; + } + + if(g_object_get_data(G_OBJECT(connection), "reaping-time")) { + g_clear_object(&account); + + return; + } + + /* Find the reaper. */ + contact_manager = purple_contact_manager_get_default(); + contact = purple_contact_manager_find_with_username(contact_manager, + account, + REAPER_BUDDY_NAME); + + if(PURPLE_IS_CONTACT(contact)) { + PurpleContactInfo *info = PURPLE_CONTACT_INFO(contact); + PurplePresence *presence = purple_contact_info_get_presence(info); + const char *format = NULL; + char *message = NULL; + + format = ngettext(tick_str, tick_plural_str, DEFAULT_REAP_TIME); + message = g_strdup_printf(format, DEFAULT_REAP_TIME); + + purple_presence_set_idle(presence, FALSE, NULL); + purple_presence_set_message(presence, message); + g_free(message); + } + + g_object_set_data(G_OBJECT(connection), "reaping-time", + GINT_TO_POINTER(DEFAULT_REAP_TIME)); + g_timeout_add_seconds(1, cb, connection); + + g_clear_object(&account); +} + +static void +purple_demo_protocol_temporary_failure_action_activate(GSimpleAction *action, + GVariant *parameter, + G_GNUC_UNUSED gpointer data) +{ + purple_demo_protocol_failure_action_activate(action, parameter, + TEMPORARY_TICK_STR, + TEMPORARY_TICK_PLURAL_STR, + purple_demo_protocol_temporary_failure_cb); +} + +static void +purple_demo_protocol_fatal_failure_action_activate(GSimpleAction *action, + GVariant *parameter, + G_GNUC_UNUSED gpointer data) +{ + purple_demo_protocol_failure_action_activate(action, parameter, + FATAL_TICK_STR, + FATAL_TICK_PLURAL_STR, + purple_demo_protocol_fatal_failure_cb); +} + +/****************************************************************************** + * Request API action implementations + *****************************************************************************/ + +static void +purple_demo_protocol_request_input_ok_cb(G_GNUC_UNUSED gpointer data, + const char *value) +{ + g_message(_("Successfully requested input from UI: %s"), value); +} + +static void +purple_demo_protocol_request_input_cancel_cb(G_GNUC_UNUSED gpointer data, + G_GNUC_UNUSED const char *value) +{ + g_message(_("UI cancelled input request")); +} + +static void +purple_demo_protocol_request_input_activate(G_GNUC_UNUSED GSimpleAction *action, + GVariant *parameter, + G_GNUC_UNUSED gpointer data) +{ + PurpleConnection *connection = NULL; + const gchar *account_id = NULL; + PurpleAccountManager *manager = NULL; + PurpleAccount *account = NULL; + static int form = 0; + gboolean multiline = FALSE, masked = FALSE; + char *secondary = NULL; + + if(!g_variant_is_of_type(parameter, G_VARIANT_TYPE_STRING)) { + g_critical("Demo failure action parameter is of incorrect type %s", + g_variant_get_type_string(parameter)); + return; + } + + account_id = g_variant_get_string(parameter, NULL); + manager = purple_account_manager_get_default(); + account = purple_account_manager_find_by_id(manager, account_id); + connection = purple_account_get_connection(account); + g_clear_object(&account); + + /* Alternate through all four combinations of {masked, multiline}. */ + masked = form % 2 == 1; + multiline = (form / 2) % 2 == 1; + form++; + secondary = g_strdup_printf(_("The input will be %s %s."), + masked ? "masked" : "unmasked", + multiline ? "multiple lines" : "single line"); + + purple_request_input(connection, _("Request Input Demo"), + _("Please input some text…"), secondary, _("default"), + multiline, masked, NULL, + _("OK"), + G_CALLBACK(purple_demo_protocol_request_input_ok_cb), + _("Cancel"), + G_CALLBACK(purple_demo_protocol_request_input_cancel_cb), + purple_request_cpar_from_connection(connection), NULL); + + g_free(secondary); +} + +static void +purple_demo_protocol_request_choice_ok_cb(G_GNUC_UNUSED gpointer data, + gpointer value) +{ + const char *text = value; + + g_message(_("Successfully requested a choice from UI: %s"), text); +} + +static void +purple_demo_protocol_request_choice_cancel_cb(G_GNUC_UNUSED gpointer data, + G_GNUC_UNUSED gpointer value) +{ + g_message(_("UI cancelled choice request")); +} + +static void +purple_demo_protocol_request_choice_activate(G_GNUC_UNUSED GSimpleAction *action, + GVariant *parameter, + G_GNUC_UNUSED gpointer data) +{ + PurpleConnection *connection = NULL; + const gchar *account_id = NULL; + PurpleAccountManager *manager = NULL; + PurpleAccount *account = NULL; + + if(!g_variant_is_of_type(parameter, G_VARIANT_TYPE_STRING)) { + g_critical("Demo failure action parameter is of incorrect type %s", + g_variant_get_type_string(parameter)); + return; + } + + account_id = g_variant_get_string(parameter, NULL); + manager = purple_account_manager_get_default(); + account = purple_account_manager_find_by_id(manager, account_id); + connection = purple_account_get_connection(account); + g_clear_object(&account); + + purple_request_choice(connection, _("Request Choice Demo"), + _("Please pick an option…"), NULL, _("foo"), + _("OK"), + G_CALLBACK(purple_demo_protocol_request_choice_ok_cb), + _("Cancel"), + G_CALLBACK(purple_demo_protocol_request_choice_cancel_cb), + purple_request_cpar_from_connection(connection), + NULL, _("foo"), "foo", _("bar"), "bar", + _("baz"), "baz", NULL); +} + +static void +purple_demo_protocol_request_action_cb(G_GNUC_UNUSED gpointer data, int action) +{ + g_message(_("Successfully requested an action from the UI: %d"), action); +} + +static void +purple_demo_protocol_request_action_activate(G_GNUC_UNUSED GSimpleAction *action, + GVariant *parameter, + G_GNUC_UNUSED gpointer data) +{ + PurpleConnection *connection = NULL; + const gchar *account_id = NULL; + PurpleAccountManager *manager = NULL; + PurpleAccount *account = NULL; + + if(!g_variant_is_of_type(parameter, G_VARIANT_TYPE_STRING)) { + g_critical("Demo failure action parameter is of incorrect type %s", + g_variant_get_type_string(parameter)); + return; + } + + account_id = g_variant_get_string(parameter, NULL); + manager = purple_account_manager_get_default(); + account = purple_account_manager_find_by_id(manager, account_id); + connection = purple_account_get_connection(account); + g_clear_object(&account); + + purple_request_action(connection, _("Request Action Demo"), + _("Please choose an action…"), NULL, 1, + purple_request_cpar_from_connection(connection), + NULL, 3, + _("foo"), purple_demo_protocol_request_action_cb, + _("bar"), purple_demo_protocol_request_action_cb, + _("baz"), purple_demo_protocol_request_action_cb); +} + +typedef struct { + gint id; + gpointer ui_handle; +} PurpleDemoProtocolWaitData; + +static gboolean +purple_demo_protocol_request_wait_pulse_cb(gpointer data) { + PurpleDemoProtocolWaitData *wait = data; + + purple_request_wait_pulse(wait->ui_handle); + + return G_SOURCE_CONTINUE; +} + +static void +purple_demo_protocol_request_wait_cancel_cb(G_GNUC_UNUSED gpointer data) { + g_message(_("UI cancelled wait request")); +} + +static void +purple_demo_protocol_request_wait_close_cb(gpointer data) { + PurpleDemoProtocolWaitData *wait = data; + + g_source_remove(wait->id); + g_free(wait); +} + +static void +purple_demo_protocol_request_wait_activate(G_GNUC_UNUSED GSimpleAction *action, + GVariant *parameter, + G_GNUC_UNUSED gpointer data) +{ + PurpleConnection *connection = NULL; + const gchar *account_id = NULL; + PurpleAccountManager *manager = NULL; + PurpleAccount *account = NULL; + PurpleDemoProtocolWaitData *wait = NULL; + + if(!g_variant_is_of_type(parameter, G_VARIANT_TYPE_STRING)) { + g_critical("Demo failure action parameter is of incorrect type %s", + g_variant_get_type_string(parameter)); + return; + } + + account_id = g_variant_get_string(parameter, NULL); + manager = purple_account_manager_get_default(); + account = purple_account_manager_find_by_id(manager, account_id); + connection = purple_account_get_connection(account); + g_clear_object(&account); + + wait = g_new0(PurpleDemoProtocolWaitData, 1); + + wait->ui_handle = purple_request_wait(connection, _("Request Wait Demo"), + _("Please wait…"), NULL, TRUE, + purple_demo_protocol_request_wait_cancel_cb, + purple_request_cpar_from_connection(connection), + wait); + + wait->id = g_timeout_add(250, purple_demo_protocol_request_wait_pulse_cb, + wait); + + purple_request_add_close_notify(wait->ui_handle, + purple_demo_protocol_request_wait_close_cb, + wait); +} + +static void +purple_demo_protocol_request_fields_ok_cb(G_GNUC_UNUSED gpointer data, + PurpleRequestPage *page) +{ + PurpleAccount *account = NULL; + PurpleRequestFieldList *field = NULL; + GList *list = NULL; + const char *tmp = NULL; + GString *info = NULL; + + info = g_string_new(_("Basic group:\n")); + + g_string_append_printf(info, _("\tString: %s\n"), + purple_request_page_get_string(page, "string")); + g_string_append_printf(info, _("\tMultiline string: %s\n"), + purple_request_page_get_string(page, + "multiline-string")); + g_string_append_printf(info, _("\tMasked string: %s\n"), + purple_request_page_get_string(page, + "masked-string")); + g_string_append_printf(info, _("\tAlphanumeric string: %s\n"), + purple_request_page_get_string(page, + "alphanumeric")); + g_string_append_printf(info, _("\tEmail string: %s\n"), + purple_request_page_get_string(page, "email")); + g_string_append_printf(info, _("\tInteger: %d\n"), + purple_request_page_get_integer(page, "int")); + g_string_append_printf(info, _("\tBoolean: %s\n"), + purple_request_page_get_bool(page, "bool") ? + _("TRUE") : _("FALSE")); + + g_string_append(info, _("Multiple-choice group:\n")); + + tmp = (const char *)purple_request_page_get_choice(page, "choice"); + g_string_append_printf(info, _("\tChoice: %s\n"), tmp); + + field = PURPLE_REQUEST_FIELD_LIST(purple_request_page_get_field(page, + "list")); + list = purple_request_field_list_get_selected(field); + if(list != NULL) { + tmp = (const char *)list->data; + } else { + tmp = _("(unset)"); + } + g_string_append_printf(info, _("\tList: %s\n"), tmp); + + field = PURPLE_REQUEST_FIELD_LIST(purple_request_page_get_field(page, + "multilist")); + list = purple_request_field_list_get_selected(field); + g_string_append(info, _("\tMulti-list: [")); + while(list != NULL) { + tmp = (const char *)list->data; + g_string_append_printf(info, "%s%s", tmp, + list->next != NULL ? ", " : ""); + list = list->next; + } + g_string_append(info, _("]\n")); + + g_string_append(info, _("Special group:\n")); + + account = purple_request_page_get_account(page, "account"); + if(PURPLE_IS_ACCOUNT(account)) { + tmp = purple_contact_info_get_name_for_display(PURPLE_CONTACT_INFO(account)); + } else { + tmp = _("(unset)"); + } + g_string_append_printf(info, _("\tAccount: %s\n"), tmp); + + g_message(_("Successfully requested fields:\n%s"), info->str); + + g_string_free(info, TRUE); +} + +static void +purple_demo_protocol_request_fields_cancel_cb(G_GNUC_UNUSED gpointer data, + G_GNUC_UNUSED PurpleRequestPage *page) +{ + g_message(_("UI cancelled field request")); +} + +static void +purple_demo_protocol_request_fields_activate(G_GNUC_UNUSED GSimpleAction *action, + GVariant *parameter, + G_GNUC_UNUSED gpointer data) +{ + PurpleConnection *connection = NULL; + const gchar *account_id = NULL; + PurpleAccountManager *manager = NULL; + PurpleAccount *account = NULL; + PurpleRequestPage *page = NULL; + PurpleRequestGroup *group = NULL; + PurpleRequestField *boolfield = NULL; + PurpleRequestField *field = NULL; + PurpleRequestFieldChoice *choice_field = NULL; + PurpleRequestFieldList *list_field = NULL; + GBytes *icon = NULL; + gconstpointer icon_data = NULL; + gsize icon_len = 0; + + if(!g_variant_is_of_type(parameter, G_VARIANT_TYPE_STRING)) { + g_critical("Demo failure action parameter is of incorrect type %s", + g_variant_get_type_string(parameter)); + return; + } + + account_id = g_variant_get_string(parameter, NULL); + manager = purple_account_manager_get_default(); + account = purple_account_manager_find_by_id(manager, account_id); + connection = purple_account_get_connection(account); + + page = purple_request_page_new(); + + /* This group will contain basic fields. */ + group = purple_request_group_new(_("Basic")); + purple_request_page_add_group(page, group); + + boolfield = purple_request_field_bool_new("bool", _("Sensitive?"), TRUE); + purple_request_field_set_tooltip(boolfield, + _("Allow modifying all fields.")); + purple_request_group_add_field(group, boolfield); + + field = purple_request_field_label_new("basic-label", + _("This group contains basic fields")); + g_object_bind_property(boolfield, "value", field, "sensitive", 0); + purple_request_group_add_field(group, field); + + field = purple_request_field_string_new("string", _("A string"), + _("default"), FALSE); + purple_request_field_set_required(field, TRUE); + g_object_bind_property(boolfield, "value", field, "sensitive", 0); + purple_request_group_add_field(group, field); + field = purple_request_field_string_new("multiline-string", + _("A multiline string"), + _("default"), TRUE); + purple_request_group_add_field(group, field); + g_object_bind_property(boolfield, "value", field, "sensitive", 0); + field = purple_request_field_string_new("masked-string", + _("A masked string"), _("default"), + FALSE); + purple_request_field_string_set_masked(PURPLE_REQUEST_FIELD_STRING(field), + TRUE); + g_object_bind_property(boolfield, "value", field, "sensitive", 0); + purple_request_group_add_field(group, field); + field = purple_request_field_string_new("alphanumeric", + _("An alphanumeric string"), + _("default"), FALSE); + purple_request_field_set_validator(field, + purple_request_field_alphanumeric_validator, + NULL, NULL); + g_object_bind_property(boolfield, "value", field, "sensitive", 0); + purple_request_group_add_field(group, field); + field = purple_request_field_string_new("email", _("An email"), + _("me@example.com"), FALSE); + purple_request_field_set_validator(field, + purple_request_field_email_validator, + NULL, NULL); + g_object_bind_property(boolfield, "value", field, "sensitive", 0); + purple_request_group_add_field(group, field); + field = purple_request_field_int_new("int", _("An integer"), 123, -42, 1337); + g_object_bind_property(boolfield, "value", field, "sensitive", 0); + purple_request_group_add_field(group, field); + + /* This group will contain fields with multiple options. */ + group = purple_request_group_new(_("Multiple")); + purple_request_page_add_group(page, group); + + field = purple_request_field_label_new("multiple-label", + _("This group contains fields with multiple options")); + g_object_bind_property(boolfield, "value", field, "sensitive", 0); + purple_request_group_add_field(group, field); + + field = purple_request_field_choice_new("choice", _("A choice"), "foo"); + choice_field = PURPLE_REQUEST_FIELD_CHOICE(field); + purple_request_field_choice_add(choice_field, _("foo"), "foo"); + purple_request_field_choice_add(choice_field, _("bar"), "bar"); + purple_request_field_choice_add(choice_field, _("baz"), "baz"); + purple_request_field_choice_add(choice_field, _("quux"), "quux"); + g_object_bind_property(boolfield, "value", field, "sensitive", 0); + purple_request_group_add_field(group, field); + + field = purple_request_field_list_new("list", _("A list")); + list_field = PURPLE_REQUEST_FIELD_LIST(field); + purple_request_field_list_add_icon(list_field, _("foo"), NULL, "foo"); + purple_request_field_list_add_icon(list_field, _("bar"), NULL, "bar"); + purple_request_field_list_add_icon(list_field, _("baz"), NULL, "baz"); + purple_request_field_list_add_icon(list_field, _("quux"), NULL, "quux"); + g_object_bind_property(boolfield, "value", field, "sensitive", 0); + purple_request_group_add_field(group, field); + + field = purple_request_field_list_new("multilist", _("A multi-select list")); + list_field = PURPLE_REQUEST_FIELD_LIST(field); + purple_request_field_list_set_multi_select(list_field, TRUE); + purple_request_field_list_add_icon(list_field, _("foo"), NULL, "foo"); + purple_request_field_list_add_icon(list_field, _("bar"), NULL, "bar"); + purple_request_field_list_add_icon(list_field, _("baz"), NULL, "baz"); + purple_request_field_list_add_icon(list_field, _("quux"), NULL, "quux"); + g_object_bind_property(boolfield, "value", field, "sensitive", 0); + purple_request_group_add_field(group, field); + + /* This group will contain specialized fields. */ + group = purple_request_group_new(_("Special")); + purple_request_page_add_group(page, group); + + field = purple_request_field_label_new("special-label", + _("This group contains specialized fields")); + g_object_bind_property(boolfield, "value", field, "sensitive", 0); + purple_request_group_add_field(group, field); + + icon = g_resource_lookup_data(purple_demo_get_resource(), + "/im/pidgin/purple/demo/icons/scalable/apps/im-purple-demo.svg", + G_RESOURCE_LOOKUP_FLAGS_NONE, NULL); + icon_data = g_bytes_get_data(icon, &icon_len); + field = purple_request_field_image_new("image", _("An image"), + icon_data, icon_len); + g_object_bind_property(boolfield, "value", field, "sensitive", 0); + purple_request_group_add_field(group, field); + g_bytes_unref(icon); + + field = purple_request_field_account_new("account", _("An account"), + account); + g_object_bind_property(boolfield, "value", field, "sensitive", 0); + purple_request_group_add_field(group, field); + + purple_request_fields(connection, _("Request Fields Demo"), + _("Please fill out these fields…"), NULL, page, + _("OK"), + G_CALLBACK(purple_demo_protocol_request_fields_ok_cb), + _("Cancel"), + G_CALLBACK(purple_demo_protocol_request_fields_cancel_cb), + purple_request_cpar_from_connection(connection), + NULL); + + g_clear_object(&account); +} + +static void +purple_demo_protocol_request_path_ok_cb(gpointer data, const char *filename) { + const char *type = data; + + g_message(_("Successfully requested %s from UI: %s"), type, filename); +} + +static void +purple_demo_protocol_request_path_cancel_cb(gpointer data) { + const char *type = data; + + g_message(_("UI cancelled %s request"), type); +} + +static void +purple_demo_protocol_request_file_activate(G_GNUC_UNUSED GSimpleAction *action, + GVariant *parameter, + G_GNUC_UNUSED gpointer data) +{ + PurpleConnection *connection = NULL; + const gchar *account_id = NULL; + PurpleAccountManager *manager = NULL; + PurpleAccount *account = NULL; + + if(!g_variant_is_of_type(parameter, G_VARIANT_TYPE_STRING)) { + g_critical("Demo failure action parameter is of incorrect type %s", + g_variant_get_type_string(parameter)); + return; + } + + account_id = g_variant_get_string(parameter, NULL); + manager = purple_account_manager_get_default(); + account = purple_account_manager_find_by_id(manager, account_id); + connection = purple_account_get_connection(account); + g_clear_object(&account); + + purple_request_file(connection, _("Request File Demo"), + "example.txt", FALSE, + G_CALLBACK(purple_demo_protocol_request_path_ok_cb), + G_CALLBACK(purple_demo_protocol_request_path_cancel_cb), + purple_request_cpar_from_connection(connection), + "file"); +} + +static void +purple_demo_protocol_request_folder_activate(G_GNUC_UNUSED GSimpleAction *action, + GVariant *parameter, + G_GNUC_UNUSED gpointer data) +{ + PurpleConnection *connection = NULL; + const gchar *account_id = NULL; + PurpleAccountManager *manager = NULL; + PurpleAccount *account = NULL; + + if(!g_variant_is_of_type(parameter, G_VARIANT_TYPE_STRING)) { + g_critical("Demo failure action parameter is of incorrect type %s", + g_variant_get_type_string(parameter)); + return; + } + + account_id = g_variant_get_string(parameter, NULL); + manager = purple_account_manager_get_default(); + account = purple_account_manager_find_by_id(manager, account_id); + connection = purple_account_get_connection(account); + g_clear_object(&account); + + purple_request_folder(connection, _("Request Folder Demo"), NULL, + G_CALLBACK(purple_demo_protocol_request_path_ok_cb), + G_CALLBACK(purple_demo_protocol_request_path_cancel_cb), + purple_request_cpar_from_connection(connection), + "folder"); +} + +/****************************************************************************** + * Contact action implementations + *****************************************************************************/ +static const gchar *contacts[] = {"Alice", "Bob", "Carlos", "Erin" }; + +static void +purple_demo_protocol_remote_add(G_GNUC_UNUSED GSimpleAction *action, + GVariant *parameter, + G_GNUC_UNUSED gpointer data) +{ + PurpleAccount *account = NULL; + PurpleAccountManager *account_manager = NULL; + PurpleAddContactRequest *request = NULL; + PurpleNotification *notification = NULL; + PurpleNotificationManager *notification_manager = NULL; + const gchar *account_id = NULL; + static guint counter = 0; + + account_id = g_variant_get_string(parameter, NULL); + account_manager = purple_account_manager_get_default(); + account = purple_account_manager_find_by_id(account_manager, account_id); + + request = purple_add_contact_request_new(account, contacts[counter]); + notification = purple_notification_new_from_add_contact_request(request); + + notification_manager = purple_notification_manager_get_default(); + purple_notification_manager_add(notification_manager, notification); + + counter++; + if(counter >= G_N_ELEMENTS(contacts)) { + counter = 0; + } + + g_clear_object(¬ification); + g_clear_object(&account); +} + +static const char *puns[] = { + "Toucan play at that game!", + "As long as it's not too much of a birden...", + "Sounds like a bit of ostrich...", + "People can't stop raven!", + "Have you heard about the bird?", +}; + +static void +purple_demo_protocol_generic_notification(G_GNUC_UNUSED GSimpleAction *action, + GVariant *parameter, + G_GNUC_UNUSED gpointer data) +{ + PurpleAccount *account = NULL; + PurpleAccountManager *account_manager = NULL; + PurpleNotification *notification = NULL; + PurpleNotificationManager *notification_manager = NULL; + const char *account_id = NULL; + static guint counter = 0; + + account_id = g_variant_get_string(parameter, NULL); + account_manager = purple_account_manager_get_default(); + account = purple_account_manager_find_by_id(account_manager, account_id); + + notification = purple_notification_new(PURPLE_NOTIFICATION_TYPE_GENERIC, + account, g_strdup(puns[counter]), + g_free); + + notification_manager = purple_notification_manager_get_default(); + purple_notification_manager_add(notification_manager, notification); + + counter++; + if(counter >= G_N_ELEMENTS(puns)) { + counter = 0; + } + + g_clear_object(¬ification); + g_clear_object(&account); +} + +/****************************************************************************** + * PurpleProtocolActions Implementation + *****************************************************************************/ +static const gchar * +purple_demo_protocol_get_prefix(G_GNUC_UNUSED PurpleProtocolActions *actions) { + return "prpl-demo"; +} + +static GActionGroup * +purple_demo_protocol_get_action_group(G_GNUC_UNUSED PurpleProtocolActions *actions, + G_GNUC_UNUSED PurpleConnection *connection) +{ + GSimpleActionGroup *group = NULL; + GActionEntry entries[] = { + { + .name = "temporary-failure", + .activate = purple_demo_protocol_temporary_failure_action_activate, + .parameter_type = "s", + }, { + .name = "fatal-failure", + .activate = purple_demo_protocol_fatal_failure_action_activate, + .parameter_type = "s", + }, { + .name = "generic-notification", + .activate = purple_demo_protocol_generic_notification, + .parameter_type = "s", + }, { + .name = "remote-add", + .activate = purple_demo_protocol_remote_add, + .parameter_type = "s", + }, { + .name = "request-input", + .activate = purple_demo_protocol_request_input_activate, + .parameter_type = "s", + }, { + .name = "request-choice", + .activate = purple_demo_protocol_request_choice_activate, + .parameter_type = "s", + }, { + .name = "request-action", + .activate = purple_demo_protocol_request_action_activate, + .parameter_type = "s", + }, { + .name = "request-wait", + .activate = purple_demo_protocol_request_wait_activate, + .parameter_type = "s", + }, { + .name = "request-fields", + .activate = purple_demo_protocol_request_fields_activate, + .parameter_type = "s", + }, { + .name = "request-file", + .activate = purple_demo_protocol_request_file_activate, + .parameter_type = "s", + }, { + .name = "request-folder", + .activate = purple_demo_protocol_request_folder_activate, + .parameter_type = "s", + } + }; + gsize nentries = G_N_ELEMENTS(entries); + + group = g_simple_action_group_new(); + g_action_map_add_action_entries(G_ACTION_MAP(group), entries, nentries, + NULL); + + return G_ACTION_GROUP(group); +} + +static GMenu * +purple_demo_protocol_get_menu(G_GNUC_UNUSED PurpleProtocolActions *actions, + G_GNUC_UNUSED PurpleConnection *connection) +{ + GMenu *menu = NULL; + GMenu *submenu = NULL; + GMenuItem *item = NULL; + + menu = g_menu_new(); + + item = g_menu_item_new(_("Trigger temporary connection failure..."), + "prpl-demo.temporary-failure"); + g_menu_item_set_attribute(item, PURPLE_MENU_ATTRIBUTE_DYNAMIC_TARGET, "s", + "account"); + g_menu_append_item(menu, item); + g_object_unref(item); + + item = g_menu_item_new(_("Trigger fatal connection failure..."), + "prpl-demo.fatal-failure"); + g_menu_item_set_attribute(item, PURPLE_MENU_ATTRIBUTE_DYNAMIC_TARGET, "s", + "account"); + g_menu_append_item(menu, item); + g_object_unref(item); + + item = g_menu_item_new(_("Trigger a generic notification"), + "prpl-demo.generic-notification"); + g_menu_item_set_attribute(item, PURPLE_MENU_ATTRIBUTE_DYNAMIC_TARGET, "s", + "account"); + g_menu_append_item(menu, item); + g_object_unref(item); + + item = g_menu_item_new(_("Trigger a contact adding you"), + "prpl-demo.remote-add"); + g_menu_item_set_attribute(item, PURPLE_MENU_ATTRIBUTE_DYNAMIC_TARGET, "s", + "account"); + g_menu_append_item(menu, item); + g_object_unref(item); + + submenu = g_menu_new(); + + item = g_menu_item_new(_("Input"), "prpl-demo.request-input"); + g_menu_item_set_attribute(item, PURPLE_MENU_ATTRIBUTE_DYNAMIC_TARGET, "s", + "account"); + g_menu_append_item(submenu, item); + g_object_unref(item); + + item = g_menu_item_new(_("Choice"), "prpl-demo.request-choice"); + g_menu_item_set_attribute(item, PURPLE_MENU_ATTRIBUTE_DYNAMIC_TARGET, "s", + "account"); + g_menu_append_item(submenu, item); + g_object_unref(item); + + item = g_menu_item_new(_("Action"), "prpl-demo.request-action"); + g_menu_item_set_attribute(item, PURPLE_MENU_ATTRIBUTE_DYNAMIC_TARGET, "s", + "account"); + g_menu_append_item(submenu, item); + g_object_unref(item); + + item = g_menu_item_new(_("Wait"), "prpl-demo.request-wait"); + g_menu_item_set_attribute(item, PURPLE_MENU_ATTRIBUTE_DYNAMIC_TARGET, "s", + "account"); + g_menu_append_item(submenu, item); + g_object_unref(item); + + item = g_menu_item_new(_("Fields"), "prpl-demo.request-fields"); + g_menu_item_set_attribute(item, PURPLE_MENU_ATTRIBUTE_DYNAMIC_TARGET, "s", + "account"); + g_menu_append_item(submenu, item); + g_object_unref(item); + + item = g_menu_item_new(_("File"), "prpl-demo.request-file"); + g_menu_item_set_attribute(item, PURPLE_MENU_ATTRIBUTE_DYNAMIC_TARGET, "s", + "account"); + g_menu_append_item(submenu, item); + g_object_unref(item); + + item = g_menu_item_new(_("Folder"), "prpl-demo.request-folder"); + g_menu_item_set_attribute(item, PURPLE_MENU_ATTRIBUTE_DYNAMIC_TARGET, "s", + "account"); + g_menu_append_item(submenu, item); + g_object_unref(item); + + g_menu_append_submenu(menu, _("Trigger requests"), G_MENU_MODEL(submenu)); + g_object_unref(submenu); + + return menu; +} + +void +purple_demo_protocol_actions_init(PurpleProtocolActionsInterface *iface) { + iface->get_prefix = purple_demo_protocol_get_prefix; + iface->get_action_group = purple_demo_protocol_get_action_group; + iface->get_menu = purple_demo_protocol_get_menu; +}