Wed, 16 Nov 2022 23:43:30 -0600
Add a contact property to PurpleAccount
This contact can be used by the protocol to store additional information about
the account, including the user id, alias, etc. When the protocol connects, it
will be responsible for adding this contact to the ContactManager once it knows
that the id is correct.
This will also be used in the future when PurpleMessage gets migrated to using
PurpleContact's as well.
Testing Done:
Connected an IRCv3 and demo accounts.
Reviewed at https://reviews.imfreedom.org/r/2064/
/* * Purple - Internet Messaging Library * Copyright (C) Pidgin Developers <devel@pidgin.im> * * 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 "purplecontact.h" #include "purpleenums.h" struct _PurpleContact { GObject parent; gchar *id; PurpleAccount *account; gchar *username; gchar *display_name; gchar *alias; GdkPixbuf *avatar; PurplePresence *presence; PurpleTags *tags; PurplePerson *person; PurpleContactPermission permission; }; enum { PROP_0, PROP_ID, PROP_ACCOUNT, PROP_USERNAME, PROP_DISPLAY_NAME, PROP_ALIAS, PROP_AVATAR, PROP_PRESENCE, PROP_TAGS, PROP_PERSON, PROP_PERMISSION, N_PROPERTIES }; static GParamSpec *properties[N_PROPERTIES] = {NULL, }; G_DEFINE_TYPE(PurpleContact, purple_contact, G_TYPE_OBJECT) /****************************************************************************** * Helpers *****************************************************************************/ static void purple_contact_set_account(PurpleContact *contact, PurpleAccount *account) { g_return_if_fail(PURPLE_IS_CONTACT(contact)); g_return_if_fail(PURPLE_IS_ACCOUNT(account)); if(g_set_object(&contact->account, account)) { g_object_notify_by_pspec(G_OBJECT(contact), properties[PROP_ACCOUNT]); } } /****************************************************************************** * GObject Implementation *****************************************************************************/ static void purple_contact_get_property(GObject *obj, guint param_id, GValue *value, GParamSpec *pspec) { PurpleContact *contact = PURPLE_CONTACT(obj); switch(param_id) { case PROP_ID: g_value_set_string(value, purple_contact_get_id(contact)); break; case PROP_ACCOUNT: g_value_set_object(value, purple_contact_get_account(contact)); break; case PROP_USERNAME: g_value_set_string(value, purple_contact_get_username(contact)); break; case PROP_DISPLAY_NAME: g_value_set_string(value, purple_contact_get_display_name(contact)); break; case PROP_ALIAS: g_value_set_string(value, purple_contact_get_alias(contact)); break; case PROP_AVATAR: g_value_set_object(value, purple_contact_get_avatar(contact)); break; case PROP_PRESENCE: g_value_set_object(value, purple_contact_get_presence(contact)); break; case PROP_TAGS: g_value_set_object(value, purple_contact_get_tags(contact)); break; case PROP_PERSON: g_value_set_object(value, purple_contact_get_person(contact)); break; case PROP_PERMISSION: g_value_set_enum(value, purple_contact_get_permission(contact)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID(obj, param_id, pspec); break; } } static void purple_contact_set_property(GObject *obj, guint param_id, const GValue *value, GParamSpec *pspec) { PurpleContact *contact = PURPLE_CONTACT(obj); switch(param_id) { case PROP_ID: purple_contact_set_id(contact, g_value_get_string(value)); break; case PROP_ACCOUNT: purple_contact_set_account(contact, g_value_get_object(value)); break; case PROP_USERNAME: purple_contact_set_username(contact, g_value_get_string(value)); break; case PROP_DISPLAY_NAME: purple_contact_set_display_name(contact, g_value_get_string(value)); break; case PROP_ALIAS: purple_contact_set_alias(contact, g_value_get_string(value)); break; case PROP_AVATAR: purple_contact_set_avatar(contact, g_value_get_object(value)); break; case PROP_PERSON: purple_contact_set_person(contact, g_value_get_object(value)); break; case PROP_PERMISSION: purple_contact_set_permission(contact, g_value_get_enum(value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID(obj, param_id, pspec); break; } } static void purple_contact_dispose(GObject *obj) { PurpleContact *contact = PURPLE_CONTACT(obj); g_clear_object(&contact->account); g_clear_object(&contact->avatar); g_clear_object(&contact->presence); g_clear_object(&contact->tags); g_clear_object(&contact->person); G_OBJECT_CLASS(purple_contact_parent_class)->dispose(obj); } static void purple_contact_finalize(GObject *obj) { PurpleContact *contact = PURPLE_CONTACT(obj); g_clear_pointer(&contact->id, g_free); g_clear_pointer(&contact->username, g_free); g_clear_pointer(&contact->display_name, g_free); g_clear_pointer(&contact->alias, g_free); G_OBJECT_CLASS(purple_contact_parent_class)->finalize(obj); } static void purple_contact_constructed(GObject *obj) { PurpleContact *contact = NULL; G_OBJECT_CLASS(purple_contact_parent_class)->constructed(obj); contact = PURPLE_CONTACT(obj); if(contact->id == NULL) { purple_contact_set_id(contact, NULL); } } static void purple_contact_init(PurpleContact *contact) { contact->tags = purple_tags_new(); contact->presence = g_object_new(PURPLE_TYPE_PRESENCE, NULL); } static void purple_contact_class_init(PurpleContactClass *klass) { GObjectClass *obj_class = G_OBJECT_CLASS(klass); obj_class->constructed = purple_contact_constructed; obj_class->dispose = purple_contact_dispose; obj_class->finalize = purple_contact_finalize; obj_class->get_property = purple_contact_get_property; obj_class->set_property = purple_contact_set_property; /** * PurpleContact:id: * * The protocol specific id for the contact. * * Since: 3.0.0 */ properties[PROP_ID] = g_param_spec_string( "id", "id", "The id of the contact", NULL, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); /** * PurpleContact:account: * * The account that this contact belongs to. * * Since: 3.0.0 */ properties[PROP_ACCOUNT] = g_param_spec_object( "account", "account", "The account this contact belongs to", PURPLE_TYPE_ACCOUNT, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS); /** * PurpleContact:username: * * The username for this contact. In rare cases this can change, like when * a user changes their "nick" on IRC which is their user name. * * Since: 3.0.0 */ properties[PROP_USERNAME] = g_param_spec_string( "username", "username", "The username of the contact", NULL, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); /** * PurpleContact:display-name: * * The display name for this contact. This is generally set by the person * the contact is representing and controlled via the protocol plugin. * * Since: 3.0.0 */ properties[PROP_DISPLAY_NAME] = g_param_spec_string( "display-name", "display-name", "The display name of the contact", NULL, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); /** * PurpleContact:alias: * * The alias for this contact. This is controlled by the libpurple user and * may be used by the protocol if it allows for aliasing. * * Since: 3.0.0 */ properties[PROP_ALIAS] = g_param_spec_string( "alias", "alias", "The alias of the contact.", NULL, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); /** * PurpleContact:avatar: * * The avatar for this contact. This is typically controlled by the protocol * and should only be read by others. * * Since: 3.0.0 */ properties[PROP_AVATAR] = g_param_spec_object( "avatar", "avatar", "The avatar of the contact", GDK_TYPE_PIXBUF, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); /** * PurpleContact:presence: * * The [class@Purple.Presence] for this contact. This is typically * controlled by the protocol and should only be read by others. * * Since: 3.0.0 */ properties[PROP_PRESENCE] = g_param_spec_object( "presence", "presence", "The presence of the contact", PURPLE_TYPE_PRESENCE, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); /** * PurpleContact:tags: * * The [class@Purple.Tags] for this contact. * * Since: 3.0.0 */ properties[PROP_TAGS] = g_param_spec_object( "tags", "tags", "The tags for the contact", PURPLE_TYPE_TAGS, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); /** * PurpleContact:person: * * The [class@Purple.Person] that this contact belongs to. * * Since: 3.0.0 */ properties[PROP_PERSON] = g_param_spec_object( "person", "person", "The person this contact belongs to.", PURPLE_TYPE_PERSON, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); /** * PurpleContact:permission: * * The permission level for the contact. * * Since: 3.0.0 */ properties[PROP_PERMISSION] = g_param_spec_enum( "permission", "permission", "The permission level of the contact", PURPLE_TYPE_CONTACT_PERMISSION, PURPLE_CONTACT_PERMISSION_UNSET, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); g_object_class_install_properties(obj_class, N_PROPERTIES, properties); } /****************************************************************************** * Public API *****************************************************************************/ PurpleContact * purple_contact_new(PurpleAccount *account, const gchar *id) { g_return_val_if_fail(PURPLE_IS_ACCOUNT(account), NULL); return g_object_new(PURPLE_TYPE_CONTACT, "account", account, "id", id, NULL); } PurpleAccount * purple_contact_get_account(PurpleContact *contact) { g_return_val_if_fail(PURPLE_IS_CONTACT(contact), NULL); return contact->account; } const gchar * purple_contact_get_id(PurpleContact *contact) { g_return_val_if_fail(PURPLE_IS_CONTACT(contact), NULL); return contact->id; } void purple_contact_set_id(PurpleContact *contact, const gchar *id) { g_return_if_fail(PURPLE_IS_CONTACT(contact)); g_free(contact->id); contact->id = g_strdup(id); g_object_notify_by_pspec(G_OBJECT(contact), properties[PROP_ID]); } const gchar * purple_contact_get_username(PurpleContact *contact) { g_return_val_if_fail(PURPLE_IS_CONTACT(contact), NULL); return contact->username; } void purple_contact_set_username(PurpleContact *contact, const gchar *username) { g_return_if_fail(PURPLE_IS_CONTACT(contact)); g_return_if_fail(username != NULL); g_free(contact->username); contact->username = g_strdup(username); g_object_notify_by_pspec(G_OBJECT(contact), properties[PROP_USERNAME]); } const gchar * purple_contact_get_display_name(PurpleContact *contact) { g_return_val_if_fail(PURPLE_IS_CONTACT(contact), NULL); return contact->display_name; } void purple_contact_set_display_name(PurpleContact *contact, const gchar *display_name) { g_return_if_fail(PURPLE_IS_CONTACT(contact)); g_free(contact->display_name); contact->display_name = g_strdup(display_name); g_object_notify_by_pspec(G_OBJECT(contact), properties[PROP_DISPLAY_NAME]); } const gchar * purple_contact_get_alias(PurpleContact *contact) { g_return_val_if_fail(PURPLE_IS_CONTACT(contact), NULL); return contact->alias; } void purple_contact_set_alias(PurpleContact *contact, const gchar *alias) { g_return_if_fail(PURPLE_IS_CONTACT(contact)); g_free(contact->alias); contact->alias = g_strdup(alias); g_object_notify_by_pspec(G_OBJECT(contact), properties[PROP_ALIAS]); } GdkPixbuf * purple_contact_get_avatar(PurpleContact *contact) { g_return_val_if_fail(PURPLE_IS_CONTACT(contact), NULL); return contact->avatar; } void purple_contact_set_avatar(PurpleContact *contact, GdkPixbuf *avatar) { g_return_if_fail(PURPLE_IS_CONTACT(contact)); if(g_set_object(&contact->avatar, avatar)) { g_object_notify_by_pspec(G_OBJECT(contact), properties[PROP_AVATAR]); } } PurplePresence * purple_contact_get_presence(PurpleContact *contact) { g_return_val_if_fail(PURPLE_IS_CONTACT(contact), NULL); return contact->presence; } PurpleTags * purple_contact_get_tags(PurpleContact *contact) { g_return_val_if_fail(PURPLE_IS_CONTACT(contact), NULL); return contact->tags; } void purple_contact_set_person(PurpleContact *contact, PurplePerson *person) { g_return_if_fail(PURPLE_IS_CONTACT(contact)); if(g_set_object(&contact->person, person)) { g_object_notify_by_pspec(G_OBJECT(contact), properties[PROP_PERSON]); } } PurplePerson * purple_contact_get_person(PurpleContact *contact) { g_return_val_if_fail(PURPLE_IS_CONTACT(contact), NULL); return contact->person; } PurpleContactPermission purple_contact_get_permission(PurpleContact *contact) { g_return_val_if_fail(PURPLE_IS_CONTACT(contact), PURPLE_CONTACT_PERMISSION_UNSET); return contact->permission; } void purple_contact_set_permission(PurpleContact *contact, PurpleContactPermission permission) { g_return_if_fail(PURPLE_IS_CONTACT(contact)); contact->permission = permission; g_object_notify_by_pspec(G_OBJECT(contact), properties[PROP_PERMISSION]); }