/* * chat_session.c * * Copyright (C) 2012 - 2014 James Booth * * 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 * along with Profanity. If not, see . * * 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. * */ #include #include #include #include #include "chat_session.h" #include "config/preferences.h" #include "log.h" #include "xmpp/xmpp.h" #define PAUSED_TIMOUT 10.0 #define INACTIVE_TIMOUT 30.0 typedef enum { CHAT_STATE_STARTED, CHAT_STATE_ACTIVE, CHAT_STATE_PAUSED, CHAT_STATE_COMPOSING, CHAT_STATE_INACTIVE, CHAT_STATE_GONE } chat_state_t; typedef struct chat_session_t { char *barejid; char *resource; gboolean send_states; chat_state_t state; GTimer *active_timer; gboolean sent; } ChatSession; static GHashTable *sessions; static ChatSession* _chat_session_new(const char * const barejid, const char * const resource, gboolean send_states) { ChatSession *new_session = malloc(sizeof(struct chat_session_t)); new_session->barejid = strdup(barejid); if (resource) { new_session->resource = strdup(resource); } else { new_session->resource = NULL; } new_session->send_states = send_states; new_session->state = CHAT_STATE_STARTED; new_session->active_timer = g_timer_new(); new_session->sent = FALSE; return new_session; } static void _chat_session_free(ChatSession *session) { if (session != NULL) { free(session->barejid); free(session->resource); if (session->active_timer != NULL) { g_timer_destroy(session->active_timer); session->active_timer = NULL; } free(session); } } void chat_sessions_init(void) { sessions = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, (GDestroyNotify)_chat_session_free); } void chat_sessions_clear(void) { if (sessions != NULL) g_hash_table_remove_all(sessions); } gboolean chat_session_exists(const char * const barejid) { ChatSession *session = g_hash_table_lookup(sessions, barejid); return (session != NULL); } char* chat_session_get_resource(const char * const barejid) { ChatSession *session = g_hash_table_lookup(sessions, barejid); assert(session != NULL); return session->resource; } gboolean chat_session_send_states(const char * const barejid) { ChatSession *session = g_hash_table_lookup(sessions, barejid); assert(session != NULL); return session->send_states; } void chat_session_on_incoming_message(const char * const barejid, const char * const resource, gboolean send_states) { ChatSession *session = g_hash_table_lookup(sessions, barejid); if (!session) { session = _chat_session_new(barejid, resource, send_states); g_hash_table_insert(sessions, strdup(barejid), session); } else if (g_strcmp0(session->resource, resource) != 0) { g_hash_table_remove(sessions, session); session = _chat_session_new(barejid, resource, send_states); g_hash_table_insert(sessions, strdup(barejid), session); } else { session->send_states = send_states; } } void chat_session_on_message_send(const char * const barejid) { ChatSession *session = g_hash_table_lookup(sessions, barejid); // if no session exists, create one with no resource, and send states if (!session) { session = _chat_session_new(barejid, NULL, TRUE); g_hash_table_insert(sessions, strdup(barejid), session); } session->state = CHAT_STATE_ACTIVE; g_timer_start(session->active_timer); session->sent = TRUE; } void chat_session_on_window_open(const char * const barejid) { ChatSession *session = g_hash_table_lookup(sessions, barejid); if (!session) { session = _chat_session_new(barejid, NULL, TRUE); g_hash_table_insert(sessions, strdup(barejid), session); } } void chat_session_on_window_close(const char * const barejid) { ChatSession *session = g_hash_table_lookup(sessions, barejid); assert(session != NULL); if (prefs_get_boolean(PREF_STATES) && session->send_states) { GString *jid = g_string_new(session->barejid); if (session->resource) { g_string_append(jid, "/"); g_string_append(jid, session->resource); } message_send_gone(jid->str); g_string_free(jid, TRUE); } } void chat_session_on_cancel(const char * const jid) { Jid *jidp = jid_create(jid); if (jidp) { ChatSession *session = g_hash_table_lookup(sessions, jidp->barejid); if (session) { session->send_states = FALSE; } } } void chat_session_on_activity(const char * const barejid) { ChatSession *session = g_hash_table_lookup(sessions, barejid); if (!session) { return; } if (session->state != CHAT_STATE_COMPOSING) { session->sent = FALSE; } session->state = CHAT_STATE_COMPOSING; g_timer_start(session->active_timer); if (!session->sent || session->state == CHAT_STATE_PAUSED) { if (prefs_get_boolean(PREF_STATES) && prefs_get_boolean(PREF_OUTTYPE) && session->send_states) { GString *jid = g_string_new(session->barejid); if (session->resource) { g_string_append(jid, "/"); g_string_append(jid, session->resource); } message_send_composing(jid->str); g_string_free(jid, TRUE); } session->sent = TRUE; } } void chat_session_on_inactivity(const char * const barejid) { ChatSession *session = g_hash_table_lookup(sessions, barejid); if (!session) { return; } if (session->active_timer != NULL) { gdouble elapsed = g_timer_elapsed(session->active_timer, NULL); if ((prefs_get_gone() != 0) && (elapsed > (prefs_get_gone() * 60.0))) { if (session->state != CHAT_STATE_GONE) { session->sent = FALSE; } session->state = CHAT_STATE_GONE; } else if (elapsed > INACTIVE_TIMOUT) { if (session->state != CHAT_STATE_INACTIVE) { session->sent = FALSE; } session->state = CHAT_STATE_INACTIVE; } else if (elapsed > PAUSED_TIMOUT) { if (session->state == CHAT_STATE_COMPOSING) { session->sent = FALSE; session->state = CHAT_STATE_PAUSED; } } } if (session->sent == FALSE) { GString *jid = g_string_new(session->barejid); if (session->resource) { g_string_append(jid, "/"); g_string_append(jid, session->resource); } if (session->state == CHAT_STATE_GONE) { if (prefs_get_boolean(PREF_STATES) && session->send_states) { message_send_gone(jid->str); } session->sent = TRUE; } else if (session->state == CHAT_STATE_INACTIVE) { if (prefs_get_boolean(PREF_STATES) && session->send_states) { message_send_inactive(jid->str); } session->sent = TRUE; } else if (session->state == CHAT_STATE_PAUSED && prefs_get_boolean(PREF_OUTTYPE)) { if (prefs_get_boolean(PREF_STATES) && session->send_states) { message_send_paused(jid->str); } session->sent = TRUE; } g_string_free(jid, TRUE); } }