* Purple is the legal property of its developers, whose names are too numerous * to list here. Please refer to the COPYRIGHT file distributed with this * 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, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA #include "conversationtypes.h" #define SEND_TYPED_TIMEOUT_SECONDS 5 /**************************************************************************/ /* PurpleIMConversation */ /**************************************************************************/ * Data specific to Instant Messages. PurpleIMTypingState typing_state; /* The current typing state. */ guint typing_timeout; /* The typing timer handle. */ time_t type_again; /* The type again time. */ guint send_typed_timeout; /* The type again timer handle. */ PurpleBuddyIcon *icon; /* The buddy icon. */ } PurpleIMConversationPrivate; static GParamSpec *im_properties[IM_PROP_LAST]; G_DEFINE_TYPE_WITH_PRIVATE(PurpleIMConversation, purple_im_conversation, PURPLE_TYPE_CONVERSATION); /**************************************************************************/ /* PurpleChatConversation */ /**************************************************************************/ * Data specific to Chats. GList *ignored; /* Ignored users. */ char *who; /* The person who set the topic. */ char *topic; /* The topic. */ int id; /* The chat ID. */ char *nick; /* Your nick in this chat. */ gboolean left; /* We left the chat and kept the window open */ GHashTable *users; /* Hash table of the users in the room. */ } PurpleChatConversationPrivate; /* Chat Property enums */ static GParamSpec *chat_properties[CHAT_PROP_LAST]; G_DEFINE_TYPE_WITH_PRIVATE(PurpleChatConversation, purple_chat_conversation, PURPLE_TYPE_CONVERSATION); /**************************************************************************/ /**************************************************************************/ * @ui_data: The UI data associated with this chat user. * Structure representing a chat user instance. * Data for "Chat Buddies" PurpleChatConversation *chat; /* The chat */ char *name; /* The chat participant's name in the char *alias; /* The chat participant's alias, if known; char *alias_key; /* A string by which this user will be sorted, or @c NULL if the user should be (This is currently always NULL. */ gboolean buddy; /* TRUE if this chat participant is on the buddy list; FALSE otherwise. */ PurpleChatUserFlags flags; /* A bitwise OR of flags for this participant, such as whether they are a channel operator. */ /* Chat User Property enums */ static GParamSpec *cu_properties[CU_PROP_LAST]; G_DEFINE_TYPE_WITH_PRIVATE(PurpleChatUser, purple_chat_user, static int purple_chat_user_compare(PurpleChatUser *a, /************************************************************************** **************************************************************************/ reset_typing_cb(gpointer data) PurpleIMConversation *im = (PurpleIMConversation *)data; purple_im_conversation_set_typing_state(im, PURPLE_IM_NOT_TYPING); purple_im_conversation_stop_typing_timeout(im); send_typed_cb(gpointer data) PurpleIMConversation *im = PURPLE_IM_CONVERSATION(data); g_return_val_if_fail(im != NULL, FALSE); gc = purple_conversation_get_connection(PURPLE_CONVERSATION(im)); name = purple_conversation_get_name(PURPLE_CONVERSATION(im)); if (gc != NULL && name != NULL) { /* We set this to 1 so that PURPLE_IM_TYPING will be sent * if the Purple user types anything else. purple_im_conversation_set_type_again(im, 1); purple_serv_send_typing(gc, name, PURPLE_IM_TYPED); purple_debug(PURPLE_DEBUG_MISC, "conversationtypes", "typed...\n"); purple_im_conversation_set_icon(PurpleIMConversation *im, PurpleBuddyIcon *icon) PurpleIMConversationPrivate *priv = NULL; g_return_if_fail(PURPLE_IS_IM_CONVERSATION(im)); priv = purple_im_conversation_get_instance_private(im); purple_buddy_icon_unref(priv->icon); priv->icon = (icon == NULL ? NULL : purple_buddy_icon_ref(icon)); g_object_notify_by_pspec(G_OBJECT(im), im_properties[IM_PROP_ICON]); purple_conversation_update(PURPLE_CONVERSATION(im), PURPLE_CONVERSATION_UPDATE_ICON); purple_im_conversation_get_icon(PurpleIMConversation *im) PurpleIMConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_IM_CONVERSATION(im), NULL); priv = purple_im_conversation_get_instance_private(im); purple_im_conversation_set_typing_state(PurpleIMConversation *im, PurpleIMTypingState state) PurpleIMConversationPrivate *priv = NULL; g_return_if_fail(PURPLE_IS_IM_CONVERSATION(im)); priv = purple_im_conversation_get_instance_private(im); name = purple_conversation_get_name(PURPLE_CONVERSATION(im)); account = purple_conversation_get_account(PURPLE_CONVERSATION(im)); if (priv->typing_state != state) priv->typing_state = state; g_object_notify_by_pspec(G_OBJECT(im), im_properties[IM_PROP_TYPING_STATE]); purple_signal_emit(purple_conversations_get_handle(), "buddy-typing", account, name); purple_signal_emit(purple_conversations_get_handle(), "buddy-typed", account, name); case PURPLE_IM_NOT_TYPING: purple_signal_emit(purple_conversations_get_handle(), "buddy-typing-stopped", account, name); purple_im_conversation_update_typing(im); purple_im_conversation_get_typing_state(PurpleIMConversation *im) PurpleIMConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_IM_CONVERSATION(im), 0); priv = purple_im_conversation_get_instance_private(im); return priv->typing_state; purple_im_conversation_start_typing_timeout(PurpleIMConversation *im, int timeout) PurpleIMConversationPrivate *priv = NULL; g_return_if_fail(PURPLE_IS_IM_CONVERSATION(im)); priv = purple_im_conversation_get_instance_private(im); if (priv->typing_timeout > 0) purple_im_conversation_stop_typing_timeout(im); priv->typing_timeout = g_timeout_add_seconds(timeout, reset_typing_cb, im); purple_im_conversation_stop_typing_timeout(PurpleIMConversation *im) PurpleIMConversationPrivate *priv = NULL; g_return_if_fail(PURPLE_IS_IM_CONVERSATION(im)); priv = purple_im_conversation_get_instance_private(im); if (priv->typing_timeout == 0) g_source_remove(priv->typing_timeout); priv->typing_timeout = 0; purple_im_conversation_get_typing_timeout(PurpleIMConversation *im) PurpleIMConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_IM_CONVERSATION(im), 0); priv = purple_im_conversation_get_instance_private(im); return priv->typing_timeout; purple_im_conversation_set_type_again(PurpleIMConversation *im, unsigned int val) PurpleIMConversationPrivate *priv = NULL; g_return_if_fail(PURPLE_IS_IM_CONVERSATION(im)); priv = purple_im_conversation_get_instance_private(im); priv->type_again = time(NULL) + val; purple_im_conversation_get_type_again(PurpleIMConversation *im) PurpleIMConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_IM_CONVERSATION(im), 0); priv = purple_im_conversation_get_instance_private(im); purple_im_conversation_start_send_typed_timeout(PurpleIMConversation *im) PurpleIMConversationPrivate *priv = NULL; g_return_if_fail(PURPLE_IS_IM_CONVERSATION(im)); priv = purple_im_conversation_get_instance_private(im); priv->send_typed_timeout = g_timeout_add_seconds(SEND_TYPED_TIMEOUT_SECONDS, purple_im_conversation_stop_send_typed_timeout(PurpleIMConversation *im) PurpleIMConversationPrivate *priv = NULL; g_return_if_fail(PURPLE_IS_IM_CONVERSATION(im)); priv = purple_im_conversation_get_instance_private(im); if (priv->send_typed_timeout == 0) g_source_remove(priv->send_typed_timeout); priv->send_typed_timeout = 0; purple_im_conversation_get_send_typed_timeout(PurpleIMConversation *im) PurpleIMConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_IM_CONVERSATION(im), 0); priv = purple_im_conversation_get_instance_private(im); return priv->send_typed_timeout; purple_im_conversation_update_typing(PurpleIMConversation *im) g_return_if_fail(PURPLE_IS_IM_CONVERSATION(im)); purple_conversation_update(PURPLE_CONVERSATION(im), PURPLE_CONVERSATION_UPDATE_TYPING); im_conversation_write_message(PurpleConversation *conv, PurpleMessage *msg) PurpleIMConversation *im = PURPLE_IM_CONVERSATION(conv); g_return_if_fail(im != NULL); g_return_if_fail(msg != NULL); is_recv = (purple_message_get_flags(msg) & PURPLE_MESSAGE_RECV); purple_im_conversation_set_typing_state(im, PURPLE_IM_NOT_TYPING); _purple_conversation_write_common(conv, msg); /************************************************************************** **************************************************************************/ /* Set method for GObject properties */ purple_im_conversation_set_property(GObject *obj, guint param_id, const GValue *value, PurpleIMConversation *im = PURPLE_IM_CONVERSATION(obj); case IM_PROP_TYPING_STATE: purple_im_conversation_set_typing_state(im, g_value_get_enum(value)); purple_im_conversation_set_icon(im, g_value_get_pointer(value)); G_OBJECT_WARN_INVALID_PROPERTY_ID(obj, param_id, pspec); /* Get method for GObject properties */ purple_im_conversation_get_property(GObject *obj, guint param_id, GValue *value, PurpleIMConversation *im = PURPLE_IM_CONVERSATION(obj); case IM_PROP_TYPING_STATE: g_value_set_enum(value, purple_im_conversation_get_typing_state(im)); g_value_set_pointer(value, purple_im_conversation_get_icon(im)); G_OBJECT_WARN_INVALID_PROPERTY_ID(obj, param_id, pspec); /* GObject initialization function */ static void purple_im_conversation_init(PurpleIMConversation *im) /* Called when done constructing */ purple_im_conversation_constructed(GObject *object) PurpleIMConversation *im = PURPLE_IM_CONVERSATION(object); G_OBJECT_CLASS(purple_im_conversation_parent_class)-> if ((icon = purple_buddy_icons_find(account, name))) purple_im_conversation_set_icon(im, icon); /* purple_im_conversation_set_icon refs the icon. */ purple_buddy_icon_unref(icon); if (purple_prefs_get_bool("/purple/logging/log_ims")) purple_conversation_set_logging(PURPLE_CONVERSATION(im), TRUE); /* GObject dispose function */ purple_im_conversation_dispose(GObject *object) PurpleIMConversationPrivate *priv = purple_im_conversation_get_instance_private( PURPLE_IM_CONVERSATION(object)); purple_buddy_icon_unref(priv->icon); G_OBJECT_CLASS(purple_im_conversation_parent_class)->dispose(object); /* GObject finalize function */ purple_im_conversation_finalize(GObject *object) PurpleIMConversation *im = PURPLE_IM_CONVERSATION(object); PurpleConnection *gc = purple_conversation_get_connection(PURPLE_CONVERSATION(im)); PurpleProtocol *protocol = NULL; const char *name = purple_conversation_get_name(PURPLE_CONVERSATION(im)); protocol = purple_connection_get_protocol(gc); if (purple_prefs_get_bool("/purple/conversations/im/send_typing")) purple_serv_send_typing(gc, name, PURPLE_IM_NOT_TYPING); purple_protocol_client_iface_convo_closed(protocol, gc, name); purple_im_conversation_stop_typing_timeout(im); purple_im_conversation_stop_send_typed_timeout(im); G_OBJECT_CLASS(purple_im_conversation_parent_class)->finalize(object); /* Class initializer function */ static void purple_im_conversation_class_init(PurpleIMConversationClass *klass) GObjectClass *obj_class = G_OBJECT_CLASS(klass); PurpleConversationClass *conv_class = PURPLE_CONVERSATION_CLASS(klass); obj_class->dispose = purple_im_conversation_dispose; obj_class->finalize = purple_im_conversation_finalize; obj_class->constructed = purple_im_conversation_constructed; obj_class->get_property = purple_im_conversation_get_property; obj_class->set_property = purple_im_conversation_set_property; conv_class->write_message = im_conversation_write_message; im_properties[IM_PROP_TYPING_STATE] = g_param_spec_enum("typing-state", "Typing state", "Status of the user's typing of a message.", PURPLE_TYPE_IM_TYPING_STATE, PURPLE_IM_NOT_TYPING, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); im_properties[IM_PROP_ICON] = g_param_spec_pointer( "icon", "Buddy icon", "The buddy icon for the IM.", G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); g_object_class_install_properties(obj_class, IM_PROP_LAST, im_properties); purple_im_conversation_new(PurpleAccount *account, const char *name) PurpleIMConversation *im; g_return_val_if_fail(PURPLE_IS_ACCOUNT(account), NULL); g_return_val_if_fail(name != NULL, NULL); /* Check if this conversation already exists. */ if ((im = purple_conversations_find_im_with_account(name, account)) != NULL) gc = purple_account_get_connection(account); g_return_val_if_fail(PURPLE_IS_CONNECTION(gc), NULL); im = g_object_new(PURPLE_TYPE_IM_CONVERSATION, /************************************************************************** **************************************************************************/ _purple_conversation_user_hash(gconstpointer data) const gchar *name = data; collated = g_utf8_collate_key(name, -1); hash = g_str_hash(collated); _purple_conversation_user_equal(gconstpointer a, gconstpointer b) return !g_utf8_collate(a, b); purple_chat_conversation_get_users(PurpleChatConversation *chat) PurpleChatConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat), NULL); priv = purple_chat_conversation_get_instance_private(chat); return g_hash_table_get_values(priv->users); purple_chat_conversation_get_users_count(PurpleChatConversation *chat) PurpleChatConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat), 0); priv = purple_chat_conversation_get_instance_private(chat); return g_hash_table_size(priv->users); purple_chat_conversation_ignore(PurpleChatConversation *chat, const char *name) PurpleChatConversationPrivate *priv = NULL; g_return_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat)); g_return_if_fail(name != NULL); priv = purple_chat_conversation_get_instance_private(chat); /* Make sure the user isn't already ignored. */ if (purple_chat_conversation_is_ignored_user(chat, name)) purple_chat_conversation_set_ignored(chat, g_list_append(priv->ignored, g_strdup(name))); purple_chat_conversation_unignore(PurpleChatConversation *chat, const char *name) PurpleChatConversationPrivate *priv = NULL; g_return_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat)); g_return_if_fail(name != NULL); priv = purple_chat_conversation_get_instance_private(chat); /* Make sure the user is actually ignored. */ if (!purple_chat_conversation_is_ignored_user(chat, name)) item = g_list_find(purple_chat_conversation_get_ignored(chat), purple_chat_conversation_get_ignored_user(chat, name)); purple_chat_conversation_set_ignored(chat, g_list_delete_link(priv->ignored, item)); purple_chat_conversation_set_ignored(PurpleChatConversation *chat, GList *ignored) PurpleChatConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat), NULL); priv = purple_chat_conversation_get_instance_private(chat); purple_chat_conversation_get_ignored(PurpleChatConversation *chat) PurpleChatConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat), NULL); priv = purple_chat_conversation_get_instance_private(chat); purple_chat_conversation_get_ignored_user(PurpleChatConversation *chat, const char *user) g_return_val_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat), NULL); g_return_val_if_fail(user != NULL, NULL); for (ignored = purple_chat_conversation_get_ignored(chat); ignored = ignored->next) { const char *ign = (const char *)ignored->data; if (!purple_utf8_strcasecmp(user, ign) || ((*ign == '+' || *ign == '%') && !purple_utf8_strcasecmp(user, ign + 1))) if ((*ign == '+' && !purple_utf8_strcasecmp(user, ign + 1)) || (*ign != '+' && !purple_utf8_strcasecmp(user, ign))) purple_chat_conversation_is_ignored_user(PurpleChatConversation *chat, const char *user) g_return_val_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat), FALSE); g_return_val_if_fail(user != NULL, FALSE); return (purple_chat_conversation_get_ignored_user(chat, user) != NULL); purple_chat_conversation_set_topic(PurpleChatConversation *chat, const char *who, const char *topic) PurpleChatConversationPrivate *priv = NULL; g_return_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat)); priv = purple_chat_conversation_get_instance_private(chat); priv->who = g_strdup(who); priv->topic = g_strdup(topic); g_object_freeze_notify(obj); g_object_notify_by_pspec(obj, chat_properties[CHAT_PROP_TOPIC_WHO]); g_object_notify_by_pspec(obj, chat_properties[CHAT_PROP_TOPIC]); g_object_thaw_notify(obj); purple_conversation_update(PURPLE_CONVERSATION(chat), PURPLE_CONVERSATION_UPDATE_TOPIC); purple_signal_emit(purple_conversations_get_handle(), "chat-topic-changed", chat, priv->who, priv->topic); purple_chat_conversation_get_topic(PurpleChatConversation *chat) PurpleChatConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat), NULL); priv = purple_chat_conversation_get_instance_private(chat); purple_chat_conversation_get_topic_who(PurpleChatConversation *chat) PurpleChatConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat), NULL); priv = purple_chat_conversation_get_instance_private(chat); purple_chat_conversation_set_id(PurpleChatConversation *chat, int id) PurpleChatConversationPrivate *priv = NULL; g_return_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat)); priv = purple_chat_conversation_get_instance_private(chat); g_object_notify_by_pspec(G_OBJECT(chat), chat_properties[CHAT_PROP_ID]); purple_chat_conversation_get_id(PurpleChatConversation *chat) PurpleChatConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat), -1); priv = purple_chat_conversation_get_instance_private(chat); chat_conversation_write_message(PurpleConversation *conv, PurpleMessage *msg) PurpleChatConversationPrivate *priv = NULL; PurpleMessageFlags flags; g_return_if_fail(PURPLE_IS_CHAT_CONVERSATION(conv)); g_return_if_fail(msg != NULL); priv = purple_chat_conversation_get_instance_private(PURPLE_CHAT_CONVERSATION(conv)); /* Don't display this if the person who wrote it is ignored. */ if (purple_message_get_author(msg) && purple_chat_conversation_is_ignored_user( PURPLE_CHAT_CONVERSATION(conv), purple_message_get_author(msg))) flags = purple_message_get_flags(msg); if (flags & PURPLE_MESSAGE_RECV) { if (purple_utf8_has_word(purple_message_get_contents(msg), priv->nick)) { flags |= PURPLE_MESSAGE_NICK; purple_message_set_flags(msg, flags); _purple_conversation_write_common(conv, msg); purple_chat_conversation_add_user(PurpleChatConversation *chat, const char *user, const char *extra_msg, PurpleChatUserFlags flags, GList *users = g_list_append(NULL, (char *)user); GList *extra_msgs = g_list_append(NULL, (char *)extra_msg); GList *flags2 = g_list_append(NULL, GINT_TO_POINTER(flags)); purple_chat_conversation_add_users(chat, users, extra_msgs, flags2, new_arrival); purple_chat_conversation_add_users(PurpleChatConversation *chat, GList *users, GList *extra_msgs, GList *flags, gboolean new_arrivals) PurpleConversation *conv; PurpleConversationUiOps *ops; PurpleChatUser *chatuser; PurpleChatConversationPrivate *priv; PurpleProtocol *protocol; g_return_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat)); g_return_if_fail(users != NULL); priv = purple_chat_conversation_get_instance_private(chat); conv = PURPLE_CONVERSATION(chat); ops = purple_conversation_get_ui_ops(conv); account = purple_conversation_get_account(conv); gc = purple_conversation_get_connection(conv); g_return_if_fail(PURPLE_IS_CONNECTION(gc)); protocol = purple_connection_get_protocol(gc); g_return_if_fail(PURPLE_IS_PROTOCOL(protocol)); while ((ul != NULL) && (fl != NULL)) { const char *user = (const char *)ul->data; const char *alias = user; PurpleChatUserFlags flag = GPOINTER_TO_INT(fl->data); const char *extra_msg = (extra_msgs ? extra_msgs->data : NULL); if(!(purple_protocol_get_options(protocol) & OPT_PROTO_UNIQUE_CHATNAME)) { if (purple_strequal(priv->nick, purple_normalize(account, user))) { const char *alias2 = purple_account_get_private_alias(account); const char *display_name = purple_connection_get_display_name(gc); if (display_name != NULL) if ((buddy = purple_blist_find_buddy(purple_connection_get_account(gc), user)) != NULL) alias = purple_buddy_get_contact_alias(buddy); quiet = GPOINTER_TO_INT(purple_signal_emit_return_1(purple_conversations_get_handle(), "chat-user-joining", chat, user, flag)) || purple_chat_conversation_is_ignored_user(chat, user); chatuser = purple_chat_user_new(chat, user, alias, flag); g_hash_table_replace(priv->users, g_strdup(purple_chat_user_get_name(chatuser)), cbuddies = g_list_prepend(cbuddies, chatuser); if (!quiet && new_arrivals) { char *alias_esc = g_markup_escape_text(alias, -1); tmp = g_strdup_printf(_("%s entered the room."), alias_esc); char *extra_msg_esc = g_markup_escape_text(extra_msg, -1); tmp = g_strdup_printf(_("%s [<I>%s</I>] entered the room."), alias_esc, extra_msg_esc); purple_conversation_write_system_message( conv, tmp, PURPLE_MESSAGE_NO_LINKIFY); purple_signal_emit(purple_conversations_get_handle(), "chat-user-joined", chat, user, flag, new_arrivals); extra_msgs = extra_msgs->next; cbuddies = g_list_sort(cbuddies, (GCompareFunc)purple_chat_user_compare); if (ops != NULL && ops->chat_add_users != NULL) ops->chat_add_users(chat, cbuddies, new_arrivals); purple_chat_conversation_rename_user(PurpleChatConversation *chat, const char *old_user, PurpleConversation *conv; PurpleConversationUiOps *ops; PurpleProtocol *protocol; PurpleChatUserFlags flags; PurpleChatConversationPrivate *priv; const char *new_alias = new_user; g_return_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat)); g_return_if_fail(old_user != NULL); g_return_if_fail(new_user != NULL); priv = purple_chat_conversation_get_instance_private(chat); conv = PURPLE_CONVERSATION(chat); ops = purple_conversation_get_ui_ops(conv); account = purple_conversation_get_account(conv); gc = purple_conversation_get_connection(conv); g_return_if_fail(PURPLE_IS_CONNECTION(gc)); protocol = purple_connection_get_protocol(gc); g_return_if_fail(PURPLE_IS_PROTOCOL(protocol)); if (purple_strequal(priv->nick, purple_normalize(account, old_user))) { /* Note this for later. */ if(!(purple_protocol_get_options(protocol) & OPT_PROTO_UNIQUE_CHATNAME)) { alias = purple_account_get_private_alias(account); const char *display_name = purple_connection_get_display_name(gc); if (display_name != NULL) new_alias = display_name; } else if (!(purple_protocol_get_options(protocol) & OPT_PROTO_UNIQUE_CHATNAME)) { if ((buddy = purple_blist_find_buddy(purple_connection_get_account(gc), new_user)) != NULL) new_alias = purple_buddy_get_contact_alias(buddy); flags = purple_chat_user_get_flags(purple_chat_conversation_find_user(chat, old_user)); cb = purple_chat_user_new(chat, new_user, new_alias, flags); g_hash_table_replace(priv->users, g_strdup(purple_chat_user_get_name(cb)), cb); if (ops != NULL && ops->chat_rename_user != NULL) ops->chat_rename_user(chat, old_user, new_user, new_alias); cb = purple_chat_conversation_find_user(chat, old_user); g_hash_table_remove(priv->users, purple_chat_user_get_name(cb)); if (purple_chat_conversation_is_ignored_user(chat, old_user)) { purple_chat_conversation_unignore(chat, old_user); purple_chat_conversation_ignore(chat, new_user); else if (purple_chat_conversation_is_ignored_user(chat, new_user)) purple_chat_conversation_unignore(chat, new_user); purple_chat_conversation_set_nick(chat, new_user); if (purple_prefs_get_bool("/purple/conversations/chat/show_nick_change") && !purple_chat_conversation_is_ignored_user(chat, new_user)) { char *escaped = g_markup_escape_text(new_user, -1); g_snprintf(tmp, sizeof(tmp), _("You are now known as %s"), escaped); const char *old_alias = old_user; const char *new_alias = new_user; if (!(purple_protocol_get_options(protocol) & OPT_PROTO_UNIQUE_CHATNAME)) { if ((buddy = purple_blist_find_buddy(purple_connection_get_account(gc), old_user)) != NULL) old_alias = purple_buddy_get_contact_alias(buddy); if ((buddy = purple_blist_find_buddy(purple_connection_get_account(gc), new_user)) != NULL) new_alias = purple_buddy_get_contact_alias(buddy); escaped = g_markup_escape_text(old_alias, -1); escaped2 = g_markup_escape_text(new_alias, -1); g_snprintf(tmp, sizeof(tmp), _("%s is now known as %s"), escaped, escaped2); purple_conversation_write_system_message(conv, tmp, PURPLE_MESSAGE_NO_LINKIFY); purple_chat_conversation_remove_user(PurpleChatConversation *chat, const char *user, const char *reason) GList *users = g_list_append(NULL, (char *)user); purple_chat_conversation_remove_users(chat, users, reason); purple_chat_conversation_remove_users(PurpleChatConversation *chat, GList *users, const char *reason) PurpleConversation *conv; PurpleProtocol *protocol; PurpleConversationUiOps *ops; PurpleChatConversationPrivate *priv; g_return_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat)); g_return_if_fail(users != NULL); priv = purple_chat_conversation_get_instance_private(chat); conv = PURPLE_CONVERSATION(chat); gc = purple_conversation_get_connection(conv); g_return_if_fail(PURPLE_IS_CONNECTION(gc)); protocol = purple_connection_get_protocol(gc); g_return_if_fail(PURPLE_IS_PROTOCOL(protocol)); ops = purple_conversation_get_ui_ops(conv); for (l = users; l != NULL; l = l->next) { const char *user = (const char *)l->data; quiet = GPOINTER_TO_INT(purple_signal_emit_return_1(purple_conversations_get_handle(), "chat-user-leaving", chat, user, reason)) | purple_chat_conversation_is_ignored_user(chat, user); cb = purple_chat_conversation_find_user(chat, user); g_hash_table_remove(priv->users, purple_chat_user_get_name(cb)); /* NOTE: Don't remove them from ignored in case they re-enter. */ const char *alias = user; if (!(purple_protocol_get_options(protocol) & OPT_PROTO_UNIQUE_CHATNAME)) { if ((buddy = purple_blist_find_buddy(purple_connection_get_account(gc), user)) != NULL) alias = purple_buddy_get_contact_alias(buddy); alias_esc = g_markup_escape_text(alias, -1); if (reason == NULL || !*reason) tmp = g_strdup_printf(_("%s left the room."), alias_esc); char *reason_esc = g_markup_escape_text(reason, -1); tmp = g_strdup_printf(_("%s left the room (%s)."), purple_conversation_write_system_message(conv, tmp, PURPLE_MESSAGE_NO_LINKIFY); purple_signal_emit(purple_conversations_get_handle(), "chat-user-left", if (ops != NULL && ops->chat_remove_users != NULL) ops->chat_remove_users(chat, users); purple_chat_conversation_clear_users(PurpleChatConversation *chat) PurpleChatConversationPrivate *priv = NULL; PurpleConversationUiOps *ops; g_return_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat)); priv = purple_chat_conversation_get_instance_private(chat); ops = purple_conversation_get_ui_ops(PURPLE_CONVERSATION(chat)); if (ops != NULL && ops->chat_remove_users != NULL) { g_hash_table_iter_init(&it, priv->users); while (g_hash_table_iter_next(&it, (gpointer*)&name, NULL)) names = g_list_prepend(names, name); ops->chat_remove_users(chat, names); g_hash_table_iter_init(&it, priv->users); while (g_hash_table_iter_next(&it, (gpointer*)&name, NULL)) { purple_signal_emit(purple_conversations_get_handle(), "chat-user-leaving", chat, name, NULL); purple_signal_emit(purple_conversations_get_handle(), "chat-user-left", chat, name, NULL); g_hash_table_remove_all(priv->users); void purple_chat_conversation_set_nick(PurpleChatConversation *chat, const char *nick) { PurpleChatConversationPrivate *priv = NULL; g_return_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat)); priv = purple_chat_conversation_get_instance_private(chat); priv->nick = g_strdup(purple_normalize( purple_conversation_get_account(PURPLE_CONVERSATION(chat)), nick)); g_object_notify_by_pspec(G_OBJECT(chat), chat_properties[CHAT_PROP_NICK]); const char *purple_chat_conversation_get_nick(PurpleChatConversation *chat) { PurpleChatConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat), NULL); priv = purple_chat_conversation_get_instance_private(chat); invite_user_to_chat(gpointer data, PurpleRequestFields *fields) PurpleConversation *conv; PurpleChatConversationPrivate *priv; const char *user, *message; priv = purple_chat_conversation_get_instance_private( PURPLE_CHAT_CONVERSATION(conv)); user = purple_request_fields_get_string(fields, "screenname"); message = purple_request_fields_get_string(fields, "message"); purple_serv_chat_invite(purple_conversation_get_connection(conv), priv->id, message, user); void purple_chat_conversation_invite_user(PurpleChatConversation *chat, const char *user, const char *message, gboolean confirm) PurpleRequestFields *fields; PurpleRequestFieldGroup *group; PurpleRequestField *field; g_return_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat)); if (!user || !*user || !message || !*message) account = purple_conversation_get_account(PURPLE_CONVERSATION(chat)); purple_serv_chat_invite(purple_account_get_connection(account), purple_chat_conversation_get_id(chat), message, user); fields = purple_request_fields_new(); group = purple_request_field_group_new(_("Invite to chat")); purple_request_fields_add_group(fields, group); field = purple_request_field_string_new("screenname", _("Buddy"), user, FALSE); purple_request_field_group_add_field(group, field); purple_request_field_set_required(field, TRUE); purple_request_field_set_type_hint(field, "screenname"); field = purple_request_field_string_new("message", _("Message"), message, FALSE); purple_request_field_group_add_field(group, field); purple_request_fields(chat, _("Invite to chat"), NULL, _("Please enter the name of the user you wish to invite, " "along with an optional invite message."), _("Invite"), G_CALLBACK(invite_user_to_chat), purple_request_cpar_from_conversation(PURPLE_CONVERSATION(chat)), purple_chat_conversation_has_user(PurpleChatConversation *chat, const char *user) g_return_val_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat), FALSE); g_return_val_if_fail(user != NULL, FALSE); return (purple_chat_conversation_find_user(chat, user) != NULL); purple_chat_conversation_leave(PurpleChatConversation *chat) PurpleChatConversationPrivate *priv = NULL; g_return_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat)); priv = purple_chat_conversation_get_instance_private(chat); if (!g_object_get_data(G_OBJECT(chat), "is-finalizing")) g_object_notify_by_pspec(G_OBJECT(chat), chat_properties[CHAT_PROP_LEFT]); purple_conversation_update(PURPLE_CONVERSATION(chat), PURPLE_CONVERSATION_UPDATE_CHATLEFT); purple_chat_conversation_has_left(PurpleChatConversation *chat) PurpleChatConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat), TRUE); priv = purple_chat_conversation_get_instance_private(chat); chat_conversation_cleanup_for_rejoin(PurpleChatConversation *chat) PurpleConversation *conv = PURPLE_CONVERSATION(chat); PurpleChatConversationPrivate *priv = purple_chat_conversation_get_instance_private(chat); account = purple_conversation_get_account(conv); purple_conversation_close_logs(conv); purple_conversation_set_logging(conv, TRUE); gc = purple_account_get_connection(account); if ((disp = purple_connection_get_display_name(gc)) != NULL) purple_chat_conversation_set_nick(chat, disp); purple_chat_conversation_set_nick(chat, purple_account_get_username(account)); purple_chat_conversation_clear_users(chat); purple_chat_conversation_set_topic(chat, NULL, NULL); g_object_notify_by_pspec(G_OBJECT(chat), chat_properties[CHAT_PROP_LEFT]); purple_conversation_update(conv, PURPLE_CONVERSATION_UPDATE_CHATLEFT); purple_chat_conversation_find_user(PurpleChatConversation *chat, const char *name) PurpleChatConversationPrivate *priv = NULL; g_return_val_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat), NULL); g_return_val_if_fail(name != NULL, NULL); priv = purple_chat_conversation_get_instance_private(chat); return g_hash_table_lookup(priv->users, name); /************************************************************************** **************************************************************************/ /* Set method for GObject properties */ purple_chat_conversation_set_property(GObject *obj, guint param_id, const GValue *value, PurpleChatConversation *chat = PURPLE_CHAT_CONVERSATION(obj); purple_chat_conversation_set_id(chat, g_value_get_int(value)); purple_chat_conversation_set_nick(chat, g_value_get_string(value)); gboolean left = g_value_get_boolean(value); purple_chat_conversation_leave(chat); G_OBJECT_WARN_INVALID_PROPERTY_ID(obj, param_id, pspec); /* Get method for GObject properties */ purple_chat_conversation_get_property(GObject *obj, guint param_id, GValue *value, PurpleChatConversation *chat = PURPLE_CHAT_CONVERSATION(obj); case CHAT_PROP_TOPIC_WHO: g_value_set_string(value, purple_chat_conversation_get_topic_who(chat)); g_value_set_string(value, purple_chat_conversation_get_topic(chat)); g_value_set_int(value, purple_chat_conversation_get_id(chat)); g_value_set_string(value, purple_chat_conversation_get_nick(chat)); g_value_set_boolean(value, purple_chat_conversation_has_left(chat)); G_OBJECT_WARN_INVALID_PROPERTY_ID(obj, param_id, pspec); /* GObject initialization function */ static void purple_chat_conversation_init(PurpleChatConversation *chat) PurpleChatConversationPrivate *priv = purple_chat_conversation_get_instance_private(chat); priv->users = g_hash_table_new_full(_purple_conversation_user_hash, _purple_conversation_user_equal, g_free, g_object_unref); /* Called when done constructing */ purple_chat_conversation_constructed(GObject *object) PurpleChatConversation *chat = PURPLE_CHAT_CONVERSATION(object); G_OBJECT_CLASS(purple_chat_conversation_parent_class)-> g_object_get(object, "account", &account, NULL); if ((disp = purple_connection_get_display_name(purple_account_get_connection(account)))) purple_chat_conversation_set_nick(chat, disp); purple_chat_conversation_set_nick(chat, purple_account_get_username(account)); if (purple_prefs_get_bool("/purple/logging/log_chats")) purple_conversation_set_logging(PURPLE_CONVERSATION(chat), TRUE); /* GObject dispose function */ purple_chat_conversation_dispose(GObject *object) PurpleChatConversationPrivate *priv = purple_chat_conversation_get_instance_private (PURPLE_CHAT_CONVERSATION(object)); g_hash_table_remove_all(priv->users); G_OBJECT_CLASS(purple_chat_conversation_parent_class)->dispose(object); /* GObject finalize function */ purple_chat_conversation_finalize(GObject *object) PurpleChatConversation *chat = PURPLE_CHAT_CONVERSATION(object); PurpleConnection *gc = purple_conversation_get_connection(PURPLE_CONVERSATION(chat)); PurpleChatConversationPrivate *priv = purple_chat_conversation_get_instance_private(chat); int chat_id = purple_chat_conversation_get_id(chat); * Close the window when the user tells us to, and let the protocol * deal with the internals on it's own time. Don't do this if the * protocol already knows it left the chat. if (!purple_chat_conversation_has_left(chat)) purple_serv_chat_leave(gc, chat_id); * If they didn't call purple_serv_got_chat_left by now, it's too late. * So we better do it for them before we destroy the thing. if (!purple_chat_conversation_has_left(chat)) purple_serv_got_chat_left(gc, chat_id); g_hash_table_destroy(priv->users); g_list_free_full(priv->ignored, g_free); G_OBJECT_CLASS(purple_chat_conversation_parent_class)->finalize(object); /* Class initializer function */ static void purple_chat_conversation_class_init(PurpleChatConversationClass *klass) GObjectClass *obj_class = G_OBJECT_CLASS(klass); PurpleConversationClass *conv_class = PURPLE_CONVERSATION_CLASS(klass); obj_class->dispose = purple_chat_conversation_dispose; obj_class->finalize = purple_chat_conversation_finalize; obj_class->constructed = purple_chat_conversation_constructed; obj_class->get_property = purple_chat_conversation_get_property; obj_class->set_property = purple_chat_conversation_set_property; conv_class->write_message = chat_conversation_write_message; chat_properties[CHAT_PROP_TOPIC_WHO] = g_param_spec_string("topic-who", "Who set the chat topic.", NULL, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); chat_properties[CHAT_PROP_TOPIC] = g_param_spec_string("topic", "Topic of the chat.", NULL, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS); chat_properties[CHAT_PROP_ID] = g_param_spec_int("chat-id", "The ID of the chat.", G_MININT, G_MAXINT, 0, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); chat_properties[CHAT_PROP_NICK] = g_param_spec_string("nick", "The nickname of the user in a chat.", NULL, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); chat_properties[CHAT_PROP_LEFT] = g_param_spec_boolean("left", "Whether the user has left the chat.", FALSE, G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); g_object_class_install_properties(obj_class, CHAT_PROP_LAST, purple_chat_conversation_new(PurpleAccount *account, const char *name) PurpleChatConversation *chat; g_return_val_if_fail(PURPLE_IS_ACCOUNT(account), NULL); g_return_val_if_fail(name != NULL, NULL); /* Check if this conversation already exists. */ if ((chat = purple_conversations_find_chat_with_account(name, account)) != NULL) if (!purple_chat_conversation_has_left(chat)) { purple_debug_warning("conversationtypes", "Trying to create " "multiple chats (%s) with the same name is deprecated and " "will be removed in libpurple 3.0.0", name); * This hack is necessary because some protocols (MSN) have unnamed chats * that all use the same name. A PurpleConversation for one of those * is only ever re-used if the user has left, so calls to * purple_conversation_new need to fall-through to creating a new * TODO 3.0.0: Remove this workaround and mandate unique names. chat_conversation_cleanup_for_rejoin(chat); gc = purple_account_get_connection(account); g_return_val_if_fail(PURPLE_IS_CONNECTION(gc), NULL); chat = g_object_new(PURPLE_TYPE_CHAT_CONVERSATION, /************************************************************************** * Chat Conversation User API **************************************************************************/ purple_chat_user_compare(PurpleChatUser *a, PurpleChatUser *b) PurpleChatUserFlags f1 = 0, f2 = 0; PurpleChatUserPrivate *priva, *privb; char *user1 = NULL, *user2 = NULL; priva = purple_chat_user_get_instance_private(a); privb = purple_chat_user_get_instance_private(b); user1 = priva->alias_key; user2 = privb->alias_key; if (user1 == NULL || user2 == NULL) { if (!(user1 == NULL && user2 == NULL)) ret = (user1 == NULL) ? -1: 1; /* sort more important users first */ ret = (f1 > f2) ? -1 : 1; } else if (priva->buddy != privb->buddy) { ret = priva->buddy ? -1 : 1; ret = purple_utf8_strcasecmp(user1, user2); purple_chat_user_get_alias(PurpleChatUser *cb) PurpleChatUserPrivate *priv; g_return_val_if_fail(PURPLE_IS_CHAT_USER(cb), NULL); priv = purple_chat_user_get_instance_private(cb); purple_chat_user_get_name(PurpleChatUser *cb) PurpleChatUserPrivate *priv; g_return_val_if_fail(PURPLE_IS_CHAT_USER(cb), NULL); priv = purple_chat_user_get_instance_private(cb); purple_chat_user_set_flags(PurpleChatUser *cb, PurpleChatUserFlags flags) PurpleConversationUiOps *ops; PurpleChatUserFlags oldflags; PurpleChatUserPrivate *priv; g_return_if_fail(PURPLE_IS_CHAT_USER(cb)); priv = purple_chat_user_get_instance_private(cb); if (flags == priv->flags) g_object_notify_by_pspec(G_OBJECT(cb), cu_properties[CU_PROP_FLAGS]); ops = purple_conversation_get_ui_ops(PURPLE_CONVERSATION(priv->chat)); if (ops != NULL && ops->chat_update_user != NULL) ops->chat_update_user(cb); purple_signal_emit(purple_conversations_get_handle(), "chat-user-flags", cb, oldflags, flags); purple_chat_user_get_flags(PurpleChatUser *cb) PurpleChatUserPrivate *priv; g_return_val_if_fail(PURPLE_IS_CHAT_USER(cb), PURPLE_CHAT_USER_NONE); priv = purple_chat_user_get_instance_private(cb); purple_chat_user_set_ui_data(PurpleChatUser *cb, gpointer ui_data) g_return_if_fail(PURPLE_IS_CHAT_USER(cb)); purple_chat_user_get_ui_data(PurpleChatUser *cb) g_return_val_if_fail(PURPLE_IS_CHAT_USER(cb), NULL); purple_chat_user_set_chat(PurpleChatUser *cb, PurpleChatConversation *chat) PurpleChatUserPrivate *priv; g_return_if_fail(PURPLE_IS_CHAT_USER(cb)); priv = purple_chat_user_get_instance_private(cb); g_object_notify_by_pspec(G_OBJECT(cb), cu_properties[CU_PROP_CHAT]); purple_chat_user_get_chat(PurpleChatUser *cb) PurpleChatUserPrivate *priv; g_return_val_if_fail(PURPLE_IS_CHAT_USER(cb), NULL); priv = purple_chat_user_get_instance_private(cb); purple_chat_user_is_buddy(PurpleChatUser *cb) PurpleChatUserPrivate *priv; g_return_val_if_fail(PURPLE_IS_CHAT_USER(cb), FALSE); priv = purple_chat_user_get_instance_private(cb); /************************************************************************** * GObject code for chat user **************************************************************************/ /* Set method for GObject properties */ purple_chat_user_set_property(GObject *obj, guint param_id, const GValue *value, PurpleChatUser *cb = PURPLE_CHAT_USER(obj); PurpleChatUserPrivate *priv = purple_chat_user_get_instance_private(cb); priv->chat = g_value_get_object(value); priv->name = g_value_dup_string(value); priv->alias = g_value_dup_string(value); priv->flags = g_value_get_flags(value); G_OBJECT_WARN_INVALID_PROPERTY_ID(obj, param_id, pspec); /* Get method for GObject properties */ purple_chat_user_get_property(GObject *obj, guint param_id, GValue *value, PurpleChatUser *cb = PURPLE_CHAT_USER(obj); g_value_set_object(value, purple_chat_user_get_chat(cb)); g_value_set_string(value, purple_chat_user_get_name(cb)); g_value_set_string(value, purple_chat_user_get_alias(cb)); g_value_set_flags(value, purple_chat_user_get_flags(cb)); G_OBJECT_WARN_INVALID_PROPERTY_ID(obj, param_id, pspec); /* GObject initialization function */ purple_chat_user_init(PurpleChatUser *user) /* Called when done constructing */ purple_chat_user_constructed(GObject *object) PurpleChatUserPrivate *priv = purple_chat_user_get_instance_private( PURPLE_CHAT_USER(object)); G_OBJECT_CLASS(purple_chat_user_parent_class)->constructed(object); account = purple_conversation_get_account(PURPLE_CONVERSATION(priv->chat)); if (purple_blist_find_buddy(account, priv->name) != NULL) /* GObject finalize function */ purple_chat_user_finalize(GObject *object) PurpleChatUser *cb = PURPLE_CHAT_USER(object); PurpleChatUserPrivate *priv = purple_chat_user_get_instance_private(cb); purple_signal_emit(purple_conversations_get_handle(), "deleting-chat-user", cb); G_OBJECT_CLASS(purple_chat_user_parent_class)->finalize(object); /* Class initializer function */ static void purple_chat_user_class_init(PurpleChatUserClass *klass) GObjectClass *obj_class = G_OBJECT_CLASS(klass); obj_class->constructed = purple_chat_user_constructed; obj_class->finalize = purple_chat_user_finalize; obj_class->get_property = purple_chat_user_get_property; obj_class->set_property = purple_chat_user_set_property; cu_properties[CU_PROP_CHAT] = g_param_spec_object("chat", "Chat", "The chat the buddy belongs to.", PURPLE_TYPE_CHAT_CONVERSATION, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | cu_properties[CU_PROP_NAME] = g_param_spec_string("name", "Name", "Name of the chat user.", NULL, G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS); cu_properties[CU_PROP_ALIAS] = g_param_spec_string("alias", "Alias", "Alias of the chat user.", NULL, G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS); cu_properties[CU_PROP_FLAGS] = g_param_spec_flags("flags", "Buddy flags", "The flags for the chat user.", PURPLE_TYPE_CHAT_USER_FLAGS, PURPLE_CHAT_USER_NONE, G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS); g_object_class_install_properties(obj_class, CU_PROP_LAST, cu_properties); purple_chat_user_new(PurpleChatConversation *chat, const char *name, const char *alias, PurpleChatUserFlags flags) g_return_val_if_fail(PURPLE_IS_CHAT_CONVERSATION(chat), NULL); g_return_val_if_fail(name != NULL, NULL); cb = g_object_new(PURPLE_TYPE_CHAT_USER,