1
0
mirror of https://github.com/profanity-im/profanity.git synced 2024-12-04 14:46:46 -05:00
profanity/src/xmpp/contact.c

448 lines
11 KiB
C
Raw Normal View History

/*
2012-05-03 20:16:37 -04:00
* contact.c
2019-11-13 06:11:05 -05:00
* vim: expandtab:ts=4:sts=4:sw=4
2012-05-03 20:16:37 -04:00
*
2019-01-22 05:31:45 -05:00
* Copyright (C) 2012 - 2019 James Booth <boothj5@gmail.com>
*
2012-05-03 20:16:37 -04:00
* This file is part of Profanity.
*
* Profanity 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 3 of the License, or
* (at your option) any later version.
*
* Profanity 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
2016-07-23 20:14:49 -04:00
* along with Profanity. If not, see <https://www.gnu.org/licenses/>.
2012-05-03 20:16:37 -04:00
*
* In addition, as a special exception, the copyright holders give permission to
* link the code of portions of this program with the OpenSSL library under
* certain conditions as described in each individual source file, and
* distribute linked combinations including the two.
*
* You must obey the GNU General Public License in all respects for all of the
* code used other than OpenSSL. If you modify file(s) with this exception, you
* may extend this exception to your version of the file(s), but you are not
* obligated to do so. If you do not wish to do so, delete this exception
* statement from your version. If you delete this exception statement from all
* source files in the program, then also delete it here.
*
2012-05-03 20:16:37 -04:00
*/
2013-02-10 13:16:06 -05:00
#include <assert.h>
2012-05-03 20:00:01 -04:00
#include <stdlib.h>
#include <string.h>
2012-05-23 19:33:23 -04:00
#include <glib.h>
2013-02-02 15:55:58 -05:00
#include "common.h"
2014-12-03 19:16:42 -05:00
#include "tools/autocomplete.h"
2016-07-24 10:43:51 -04:00
#include "xmpp/resource.h"
#include "xmpp/contact.h"
2013-02-02 15:55:58 -05:00
2012-05-03 20:00:01 -04:00
struct p_contact_t {
2013-02-09 18:50:41 -05:00
char *barejid;
gchar *barejid_collate_key;
2012-05-03 20:00:01 -04:00
char *name;
gchar *name_collate_key;
GSList *groups;
2012-10-28 16:52:30 -04:00
char *subscription;
char *offline_message;
gboolean pending_out;
2012-12-08 22:07:33 -05:00
GDateTime *last_activity;
2013-02-10 07:36:58 -05:00
GHashTable *available_resources;
2014-12-03 19:16:42 -05:00
Autocomplete resource_ac;
2012-05-03 20:00:01 -04:00
};
2012-07-24 18:19:48 -04:00
PContact
2015-10-25 20:52:33 -04:00
p_contact_new(const char *const barejid, const char *const name,
GSList *groups, const char *const subscription,
const char *const offline_message, gboolean pending_out)
2012-05-03 20:00:01 -04:00
{
PContact contact = malloc(sizeof(struct p_contact_t));
2013-02-09 18:50:41 -05:00
contact->barejid = strdup(barejid);
contact->barejid_collate_key = g_utf8_collate_key(contact->barejid, -1);
2012-05-03 20:00:01 -04:00
2015-05-04 18:26:57 -04:00
if (name) {
2012-10-28 16:52:30 -04:00
contact->name = strdup(name);
contact->name_collate_key = g_utf8_collate_key(contact->name, -1);
2012-10-28 16:52:30 -04:00
} else {
contact->name = NULL;
contact->name_collate_key = NULL;
2012-10-28 16:52:30 -04:00
}
contact->groups = groups;
2015-05-04 18:26:57 -04:00
if (subscription)
2012-10-28 17:16:22 -04:00
contact->subscription = strdup(subscription);
else
contact->subscription = strdup("none");
2015-05-04 18:26:57 -04:00
if (offline_message)
contact->offline_message = strdup(offline_message);
else
contact->offline_message = NULL;
contact->pending_out = pending_out;
2012-12-08 22:07:33 -05:00
contact->last_activity = NULL;
2013-02-10 07:36:58 -05:00
contact->available_resources = g_hash_table_new_full(g_str_hash, g_str_equal, free,
2013-02-09 20:20:07 -05:00
(GDestroyNotify)resource_destroy);
2013-02-09 21:17:22 -05:00
2014-12-03 19:16:42 -05:00
contact->resource_ac = autocomplete_new();
2013-02-09 21:17:22 -05:00
return contact;
}
void
2015-10-25 20:52:33 -04:00
p_contact_set_name(const PContact contact, const char *const name)
{
FREE_SET_NULL(contact->name);
FREE_SET_NULL(contact->name_collate_key);
2015-05-04 18:26:57 -04:00
if (name) {
contact->name = strdup(name);
contact->name_collate_key = g_utf8_collate_key(contact->name, -1);
}
}
void
p_contact_set_groups(const PContact contact, GSList *groups)
{
2015-05-04 18:26:57 -04:00
if (contact->groups) {
g_slist_free_full(contact->groups, g_free);
contact->groups = NULL;
}
contact->groups = groups;
}
gboolean
2015-10-25 20:52:33 -04:00
p_contact_in_group(const PContact contact, const char *const group)
{
GSList *groups = contact->groups;
2015-05-04 18:26:57 -04:00
while (groups) {
if (strcmp(groups->data, group) == 0) {
return TRUE;
}
groups = g_slist_next(groups);
}
return FALSE;
}
2015-10-25 20:52:33 -04:00
GSList*
p_contact_groups(const PContact contact)
{
return contact->groups;
}
gboolean
2015-10-25 20:52:33 -04:00
p_contact_remove_resource(PContact contact, const char *const resource)
2013-02-09 21:17:22 -05:00
{
2014-12-03 19:16:42 -05:00
gboolean result = g_hash_table_remove(contact->available_resources, resource);
autocomplete_remove(contact->resource_ac, resource);
return result;
2012-05-17 20:33:40 -04:00
}
2012-07-24 18:19:48 -04:00
void
p_contact_free(PContact contact)
2012-05-03 20:00:01 -04:00
{
2015-05-04 18:26:57 -04:00
if (contact) {
free(contact->barejid);
free(contact->barejid_collate_key);
free(contact->name);
free(contact->name_collate_key);
free(contact->subscription);
free(contact->offline_message);
2015-05-04 18:26:57 -04:00
if (contact->groups) {
g_slist_free_full(contact->groups, g_free);
}
2012-10-28 17:16:22 -04:00
2015-05-04 18:26:57 -04:00
if (contact->last_activity) {
g_date_time_unref(contact->last_activity);
}
g_hash_table_destroy(contact->available_resources);
2014-12-03 19:16:42 -05:00
autocomplete_free(contact->resource_ac);
free(contact);
2012-12-08 22:07:33 -05:00
}
2012-05-03 20:00:01 -04:00
}
2015-10-25 20:52:33 -04:00
const char*
2013-02-09 18:50:41 -05:00
p_contact_barejid(const PContact contact)
2012-10-28 16:52:30 -04:00
{
2013-02-09 18:50:41 -05:00
return contact->barejid;
2012-10-28 16:52:30 -04:00
}
2015-10-25 20:52:33 -04:00
const char*
p_contact_barejid_collate_key(const PContact contact)
{
return contact->barejid_collate_key;
}
2015-10-25 20:52:33 -04:00
const char*
2012-07-24 18:19:48 -04:00
p_contact_name(const PContact contact)
2012-05-03 20:00:01 -04:00
{
return contact->name;
}
2015-10-25 20:52:33 -04:00
const char*
p_contact_name_collate_key(const PContact contact)
{
return contact->name_collate_key;
}
2015-10-25 20:52:33 -04:00
const char*
2013-08-31 10:07:05 -04:00
p_contact_name_or_jid(const PContact contact)
{
2015-05-04 18:26:57 -04:00
if (contact->name) {
2013-08-31 10:07:05 -04:00
return contact->name;
} else {
return contact->barejid;
}
}
2015-10-25 20:52:33 -04:00
char*
p_contact_create_display_string(const PContact contact, const char *const resource)
2013-10-06 19:16:58 -04:00
{
GString *result_str = g_string_new("");
2014-01-04 19:43:37 -05:00
// use nickname if exists
const char *display_name = p_contact_name_or_jid(contact);
g_string_append(result_str, display_name);
2013-10-06 19:16:58 -04:00
// add resource if not default provided by profanity
if (strcmp(resource, "__prof_default") != 0) {
g_string_append(result_str, " (");
g_string_append(result_str, resource);
g_string_append(result_str, ")");
}
char *result = result_str->str;
g_string_free(result_str, FALSE);
return result;
}
2015-10-25 20:52:33 -04:00
static Resource*
_highest_presence(Resource *first, Resource *second)
{
if (first->presence == RESOURCE_CHAT) {
return first;
} else if (second->presence == RESOURCE_CHAT) {
return second;
} else if (first->presence == RESOURCE_ONLINE) {
return first;
} else if (second->presence == RESOURCE_ONLINE) {
return second;
} else if (first->presence == RESOURCE_AWAY) {
return first;
} else if (second->presence == RESOURCE_AWAY) {
return second;
} else if (first->presence == RESOURCE_XA) {
return first;
} else if (second->presence == RESOURCE_XA) {
return second;
} else {
return first;
}
}
2015-10-25 20:52:33 -04:00
Resource*
_get_most_available_resource(PContact contact)
2012-05-03 20:00:01 -04:00
{
// find resource with highest priority, if more than one,
// use highest availability, in the following order:
// chat
// online
// away
// xa
// dnd
GList *resources = g_hash_table_get_values(contact->available_resources);
GList *curr = resources;
Resource *current = curr->data;
Resource *highest = current;
curr = g_list_next(curr);
2015-05-04 18:26:57 -04:00
while (curr) {
current = curr->data;
// priority is same as current highest, choose presence
if (current->priority == highest->priority) {
highest = _highest_presence(highest, current);
// priority higher than current highest, set new presence
} else if (current->priority > highest->priority) {
highest = current;
}
curr = g_list_next(curr);
}
g_list_free(resources);
return highest;
2012-05-03 20:00:01 -04:00
}
2015-10-25 20:52:33 -04:00
const char*
p_contact_presence(const PContact contact)
2012-05-03 20:00:01 -04:00
{
assert(contact != NULL);
// no available resources, offline
if (g_hash_table_size(contact->available_resources) == 0) {
return "offline";
}
Resource *resource = _get_most_available_resource(contact);
return string_from_resource_presence(resource->presence);
}
2015-10-25 20:52:33 -04:00
const char*
p_contact_status(const PContact contact)
{
assert(contact != NULL);
// no available resources, use offline message
if (g_hash_table_size(contact->available_resources) == 0) {
return contact->offline_message;
}
Resource *resource = _get_most_available_resource(contact);
return resource->status;
2012-05-03 20:00:01 -04:00
}
2012-05-23 18:53:10 -04:00
2015-10-25 20:52:33 -04:00
const char*
2012-10-28 17:16:22 -04:00
p_contact_subscription(const PContact contact)
{
return contact->subscription;
}
gboolean
p_contact_subscribed(const PContact contact)
{
if (contact->subscription == NULL) {
return FALSE;
} else if (strcmp(contact->subscription, "to") == 0) {
return TRUE;
} else if (strcmp(contact->subscription, "both") == 0) {
return TRUE;
} else {
return FALSE;
}
}
2015-10-25 20:52:33 -04:00
Resource*
p_contact_get_resource(const PContact contact, const char *const resource)
{
return g_hash_table_lookup(contact->available_resources, resource);
}
gboolean
p_contact_pending_out(const PContact contact)
{
return contact->pending_out;
}
2015-10-25 20:52:33 -04:00
GDateTime*
2012-12-08 22:07:33 -05:00
p_contact_last_activity(const PContact contact)
{
return contact->last_activity;
}
2015-10-25 20:52:33 -04:00
GList*
p_contact_get_available_resources(const PContact contact)
{
assert(contact != NULL);
2014-11-12 20:01:41 -05:00
GList *resources = g_hash_table_get_values(contact->available_resources);
GList *ordered = NULL;
GList *curr_resource = resources;
while (curr_resource) {
2014-11-12 20:19:20 -05:00
Resource *resource = curr_resource->data;
2014-11-12 20:01:41 -05:00
ordered = g_list_insert_sorted(ordered, resource, (GCompareFunc)resource_compare_availability);
curr_resource = g_list_next(curr_resource);
}
g_list_free(resources);
2014-11-12 20:01:41 -05:00
return ordered;
}
gboolean
p_contact_is_available(const PContact contact)
{
// no available resources, unavailable
if (g_hash_table_size(contact->available_resources) == 0) {
return FALSE;
}
// if most available resource is CHAT or ONLINE, available
Resource *most_available = _get_most_available_resource(contact);
if ((most_available->presence == RESOURCE_ONLINE) ||
(most_available->presence == RESOURCE_CHAT)) {
return TRUE;
} else {
return FALSE;
}
}
gboolean
p_contact_has_available_resource(const PContact contact)
{
return (g_hash_table_size(contact->available_resources) > 0);
}
2012-10-29 17:44:33 -04:00
void
p_contact_set_presence(const PContact contact, Resource *resource)
2012-10-29 17:44:33 -04:00
{
g_hash_table_replace(contact->available_resources, strdup(resource->name), resource);
2015-02-06 17:03:40 -05:00
autocomplete_add(contact->resource_ac, resource->name);
2012-10-29 17:44:33 -04:00
}
2012-11-27 18:43:32 -05:00
void
2015-10-25 20:52:33 -04:00
p_contact_set_subscription(const PContact contact, const char *const subscription)
2012-11-27 18:43:32 -05:00
{
FREE_SET_NULL(contact->subscription);
2015-05-04 18:26:57 -04:00
if (subscription) {
2012-11-27 18:43:32 -05:00
contact->subscription = strdup(subscription);
}
}
void
p_contact_set_pending_out(const PContact contact, gboolean pending_out)
{
contact->pending_out = pending_out;
}
2012-12-08 22:07:33 -05:00
void
p_contact_set_last_activity(const PContact contact, GDateTime *last_activity)
2012-05-23 18:53:10 -04:00
{
2015-05-04 18:26:57 -04:00
if (contact->last_activity) {
2012-12-08 22:07:33 -05:00
g_date_time_unref(contact->last_activity);
contact->last_activity = NULL;
}
2012-05-23 18:53:10 -04:00
2015-05-04 18:26:57 -04:00
if (last_activity) {
2012-12-08 22:07:33 -05:00
contact->last_activity = g_date_time_ref(last_activity);
}
2012-05-23 18:53:10 -04:00
}
2014-12-03 19:16:42 -05:00
Autocomplete
p_contact_resource_ac(const PContact contact)
{
return contact->resource_ac;
}
void
p_contact_resource_ac_reset(const PContact contact)
{
autocomplete_reset(contact->resource_ac);
2015-10-25 20:52:33 -04:00
}