/* * mucwin.c * vim: expandtab:ts=4:sts=4:sw=4 * * Copyright (C) 2012 - 2019 James Booth * Copyright (C) 2019 - 2023 Michael Vetter * * 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 "config.h" #include "ui.h" #include #include #include #include "log.h" #include "config/preferences.h" #include "plugins/plugins.h" #include "ui/window.h" #include "ui/win_types.h" #include "ui/window_list.h" #ifdef HAVE_OMEMO #include "omemo/omemo.h" #endif static void _mucwin_set_last_message(ProfMucWin* mucwin, const char* const id, const char* const message); ProfMucWin* mucwin_new(const char* const barejid) { ProfWin* window = wins_new_muc(barejid); ProfMucWin* mucwin = (ProfMucWin*)window; mucwin->last_msg_timestamp = NULL; #ifdef HAVE_OMEMO if (muc_anonymity_type(mucwin->roomjid) == MUC_ANONYMITY_TYPE_NONANONYMOUS && omemo_automatic_start(barejid)) { omemo_start_muc_sessions(barejid); mucwin->is_omemo = TRUE; } #endif // Force redraw here to show correct offline users; before this point muc_members returns a wrong list ui_redraw_all_room_rosters(); return mucwin; } void mucwin_role_change(ProfMucWin* mucwin, const char* const role, const char* const actor, const char* const reason) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_print(window, THEME_ROOMINFO, "!", "Your role has been changed to: %s", role); if (actor) { win_append(window, THEME_ROOMINFO, ", by: %s", actor); } if (reason) { win_append(window, THEME_ROOMINFO, ", reason: %s", reason); } win_appendln(window, THEME_ROOMINFO, ""); } void mucwin_affiliation_change(ProfMucWin* mucwin, const char* const affiliation, const char* const actor, const char* const reason) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_print(window, THEME_ROOMINFO, "!", "Your affiliation has been changed to: %s", affiliation); if (actor) { win_append(window, THEME_ROOMINFO, ", by: %s", actor); } if (reason) { win_append(window, THEME_ROOMINFO, ", reason: %s", reason); } win_appendln(window, THEME_ROOMINFO, ""); } void mucwin_role_and_affiliation_change(ProfMucWin* mucwin, const char* const role, const char* const affiliation, const char* const actor, const char* const reason) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_print(window, THEME_ROOMINFO, "!", "Your role and affiliation have been changed, role: %s, affiliation: %s", role, affiliation); if (actor) { win_append(window, THEME_ROOMINFO, ", by: %s", actor); } if (reason) { win_append(window, THEME_ROOMINFO, ", reason: %s", reason); } win_appendln(window, THEME_ROOMINFO, ""); } void mucwin_occupant_role_change(ProfMucWin* mucwin, const char* const nick, const char* const role, const char* const actor, const char* const reason) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_print(window, THEME_ROOMINFO, "!", "%s's role has been changed to: %s", nick, role); if (actor) { win_append(window, THEME_ROOMINFO, ", by: %s", actor); } if (reason) { win_append(window, THEME_ROOMINFO, ", reason: %s", reason); } win_appendln(window, THEME_ROOMINFO, ""); } void mucwin_occupant_affiliation_change(ProfMucWin* mucwin, const char* const nick, const char* const affiliation, const char* const actor, const char* const reason) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_print(window, THEME_ROOMINFO, "!", "%s's affiliation has been changed to: %s", nick, affiliation); if (actor) { win_append(window, THEME_ROOMINFO, ", by: %s", actor); } if (reason) { win_append(window, THEME_ROOMINFO, ", reason: %s", reason); } win_appendln(window, THEME_ROOMINFO, ""); } void mucwin_occupant_role_and_affiliation_change(ProfMucWin* mucwin, const char* const nick, const char* const role, const char* const affiliation, const char* const actor, const char* const reason) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_print(window, THEME_ROOMINFO, "!", "%s's role and affiliation have been changed, role: %s, affiliation: %s", nick, role, affiliation); if (actor) { win_append(window, THEME_ROOMINFO, ", by: %s", actor); } if (reason) { win_append(window, THEME_ROOMINFO, ", reason: %s", reason); } win_appendln(window, THEME_ROOMINFO, ""); } void mucwin_room_info_error(ProfMucWin* mucwin, const char* const error) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_println(window, THEME_DEFAULT, "!", "Room info request failed: %s", error); win_println(window, THEME_DEFAULT, "-", ""); } void mucwin_room_disco_info(ProfMucWin* mucwin, GSList* identities, GSList* features) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; if ((identities && (g_slist_length(identities) > 0)) || (features && (g_slist_length(features) > 0))) { if (identities) { win_println(window, THEME_DEFAULT, "!", "Identities:"); } while (identities) { DiscoIdentity* identity = identities->data; // name, type, category GString* identity_str = g_string_new(" "); if (identity->name) { identity_str = g_string_append(identity_str, identity->name); identity_str = g_string_append(identity_str, " "); } if (identity->type) { identity_str = g_string_append(identity_str, identity->type); identity_str = g_string_append(identity_str, " "); } if (identity->category) { identity_str = g_string_append(identity_str, identity->category); } win_println(window, THEME_DEFAULT, "!", "%s", identity_str->str); g_string_free(identity_str, TRUE); identities = g_slist_next(identities); } if (features) { win_println(window, THEME_DEFAULT, "!", "Features:"); } while (features) { win_println(window, THEME_DEFAULT, "!", " %s", features->data); features = g_slist_next(features); } win_println(window, THEME_DEFAULT, "-", ""); } } void mucwin_roster(ProfMucWin* mucwin, GList* roster, const char* const presence) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; if ((roster == NULL) || (g_list_length(roster) == 0)) { if (presence == NULL) { win_println(window, THEME_ROOMINFO, "!", "Room is empty."); } else { win_println(window, THEME_ROOMINFO, "!", "No occupants %s.", presence); } } else { int length = g_list_length(roster); if (presence == NULL) { win_print(window, THEME_ROOMINFO, "!", "%d occupants: ", length); } else { win_print(window, THEME_ROOMINFO, "!", "%d %s: ", length, presence); } while (roster) { Occupant* occupant = roster->data; const char* presence_str = string_from_resource_presence(occupant->presence); theme_item_t presence_colour = theme_main_presence_attrs(presence_str); win_append(window, presence_colour, "%s", occupant->nick); if (roster->next) { win_append(window, THEME_DEFAULT, ", "); } roster = g_list_next(roster); } win_appendln(window, THEME_ONLINE, ""); } } void mucwin_occupant_offline(ProfMucWin* mucwin, const char* const nick) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_println(window, THEME_OFFLINE, "!", "<- %s has left the room.", nick); } void mucwin_occupant_kicked(ProfMucWin* mucwin, const char* const nick, const char* const actor, const char* const reason) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; GString* message = g_string_new(nick); g_string_append(message, " has been kicked from the room"); if (actor) { g_string_append(message, " by "); g_string_append(message, actor); } if (reason) { g_string_append(message, ", reason: "); g_string_append(message, reason); } win_println(window, THEME_OFFLINE, "!", "<- %s", message->str); g_string_free(message, TRUE); } void mucwin_occupant_banned(ProfMucWin* mucwin, const char* const nick, const char* const actor, const char* const reason) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; GString* message = g_string_new(nick); g_string_append(message, " has been banned from the room"); if (actor) { g_string_append(message, " by "); g_string_append(message, actor); } if (reason) { g_string_append(message, ", reason: "); g_string_append(message, reason); } win_println(window, THEME_OFFLINE, "!", "<- %s", message->str); g_string_free(message, TRUE); } void mucwin_occupant_online(ProfMucWin* mucwin, const char* const nick, const char* const role, const char* const affiliation, const char* const show, const char* const status) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_print(window, THEME_ONLINE, "!", "-> %s has joined the room", nick); if (prefs_get_boolean(PREF_MUC_PRIVILEGES)) { if (role) { win_append(window, THEME_ONLINE, ", role: %s", role); } if (affiliation) { win_append(window, THEME_ONLINE, ", affiliation: %s", affiliation); } } win_appendln(window, THEME_ROOMINFO, ""); } void mucwin_occupant_presence(ProfMucWin* mucwin, const char* const nick, const char* const show, const char* const status) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_show_status_string(window, nick, show, status, NULL, "++", "online"); } void mucwin_occupant_nick_change(ProfMucWin* mucwin, const char* const old_nick, const char* const nick) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_println(window, THEME_THEM, "!", "** %s is now known as %s", old_nick, nick); } void mucwin_nick_change(ProfMucWin* mucwin, const char* const nick) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_println(window, THEME_ME, "!", "** You are now known as %s", nick); } void mucwin_history(ProfMucWin* mucwin, const ProfMessage* const message) { assert(mucwin != NULL); char* nick = message->from_jid->resourcepart; char* mynick = muc_nick(mucwin->roomjid); GSList* mentions = get_mentions(prefs_get_boolean(PREF_NOTIFY_MENTION_WHOLE_WORD), prefs_get_boolean(PREF_NOTIFY_MENTION_CASE_SENSITIVE), message->plain, mynick); GList* triggers = prefs_message_get_triggers(message->plain); mucwin_incoming_msg(mucwin, message, mentions, triggers, FALSE); g_slist_free(mentions); g_list_free_full(triggers, free); plugins_on_room_history_message(mucwin->roomjid, nick, message->plain, message->timestamp); } static void _mucwin_print_mention(ProfWin* window, const char* const message, const char* const from, const char* const mynick, GSList* mentions, const char* const ch, int flags) { int last_pos = 0; int pos; GSList* curr = mentions; glong mynick_len = g_utf8_strlen(mynick, -1); while (curr) { pos = GPOINTER_TO_INT(curr->data); char* before_str = g_utf8_substring(message, last_pos, pos); if (last_pos == 0 && strncmp(before_str, "/me ", 4) == 0) { win_print_them(window, THEME_ROOMMENTION, ch, flags, ""); win_append_highlight(window, THEME_ROOMMENTION, "*%s ", from); win_append_highlight(window, THEME_ROOMMENTION, "%s", before_str + 4); } else { // print time and nick only once at beginning of the line if (last_pos == 0) { win_print_them(window, THEME_ROOMMENTION, ch, flags, from); } win_append_highlight(window, THEME_ROOMMENTION, "%s", before_str); } g_free(before_str); char* mynick_str = g_utf8_substring(message, pos, pos + mynick_len); win_append_highlight(window, THEME_ROOMMENTION_TERM, "%s", mynick_str); g_free(mynick_str); last_pos = pos + mynick_len; curr = g_slist_next(curr); } glong message_len = g_utf8_strlen(message, -1); if (last_pos < message_len) { // get tail without allocating a new string char* rest = g_utf8_offset_to_pointer(message, last_pos); win_appendln_highlight(window, THEME_ROOMMENTION, "%s", rest); } else { win_appendln_highlight(window, THEME_ROOMMENTION, ""); } } gint _cmp_trigger_weight(gconstpointer a, gconstpointer b) { int alen = strlen((char*)a); int blen = strlen((char*)b); if (alen > blen) return -1; if (alen < blen) return 1; return 0; } static void _mucwin_print_triggers(ProfWin* window, const char* const message, GList* triggers) { GList* weighted_triggers = NULL; GList* curr = triggers; while (curr) { weighted_triggers = g_list_insert_sorted(weighted_triggers, curr->data, (GCompareFunc)_cmp_trigger_weight); curr = g_list_next(curr); } char* message_lower = g_utf8_strdown(message, -1); // find earliest trigger in message int first_trigger_pos = -1; int first_trigger_len = -1; curr = weighted_triggers; while (curr) { char* trigger_lower = g_utf8_strdown(curr->data, -1); char* trigger_ptr = g_strstr_len(message_lower, -1, trigger_lower); // not found, try next if (trigger_ptr == NULL) { curr = g_list_next(curr); continue; } // found, replace vars if earlier than previous int trigger_pos = trigger_ptr - message_lower; if (first_trigger_pos == -1 || trigger_pos < first_trigger_pos) { first_trigger_pos = trigger_pos; first_trigger_len = strlen(trigger_lower); } g_free(trigger_lower); curr = g_list_next(curr); } g_free(message_lower); g_list_free(weighted_triggers); // no triggers found if (first_trigger_pos == -1) { win_appendln_highlight(window, THEME_ROOMTRIGGER, "%s", message); } else { if (first_trigger_pos > 0) { char message_section[strlen(message) + 1]; int i = 0; while (i < first_trigger_pos) { message_section[i] = message[i]; i++; } message_section[i] = '\0'; win_append_highlight(window, THEME_ROOMTRIGGER, "%s", message_section); } char trigger_section[first_trigger_len + 1]; int i = 0; while (i < first_trigger_len) { trigger_section[i] = message[first_trigger_pos + i]; i++; } trigger_section[i] = '\0'; if (first_trigger_pos + first_trigger_len < strlen(message)) { win_append_highlight(window, THEME_ROOMTRIGGER_TERM, "%s", trigger_section); _mucwin_print_triggers(window, &message[first_trigger_pos + first_trigger_len], triggers); } else { win_appendln_highlight(window, THEME_ROOMTRIGGER_TERM, "%s", trigger_section); } } } void mucwin_outgoing_msg(ProfMucWin* mucwin, const char* const message, const char* const id, prof_enc_t enc_mode, const char* const replace_id) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; char* mynick = muc_nick(mucwin->roomjid); // displayed message char char* ch; if (mucwin->message_char) { ch = strdup(mucwin->message_char); } else if (enc_mode == PROF_MSG_ENC_OTR) { ch = prefs_get_otr_char(); } else if (enc_mode == PROF_MSG_ENC_PGP) { ch = prefs_get_pgp_char(); } else if (enc_mode == PROF_MSG_ENC_OMEMO) { ch = prefs_get_omemo_char(); } else if (enc_mode == PROF_MSG_ENC_OX) { ch = prefs_get_omemo_char(); } else { ch = strdup("-"); } win_print_outgoing_muc_msg(window, ch, mynick, id, replace_id, message); free(ch); // save last id and message for LMC if (id) { _mucwin_set_last_message(mucwin, id, message); } wins_add_quotes_ac(window, message, FALSE); } void mucwin_incoming_msg(ProfMucWin* mucwin, const ProfMessage* const message, GSList* mentions, GList* triggers, gboolean filter_reflection) { assert(mucwin != NULL); int flags = 0; if (message->plain == NULL) { log_error("mucwin_incoming_msg: Message with no plain field from: %s", message->from_jid); return; } if (filter_reflection && message_is_sent_by_us(message, TRUE)) { /* Ignore reflection messages */ return; } if (!message->trusted) { flags |= UNTRUSTED; } ProfWin* window = (ProfWin*)mucwin; char* mynick = muc_nick(mucwin->roomjid); char* ch; if (mucwin->message_char) { ch = strdup(mucwin->message_char); } else if (message->enc == PROF_MSG_ENC_OTR) { ch = prefs_get_otr_char(); } else if (message->enc == PROF_MSG_ENC_PGP) { ch = prefs_get_pgp_char(); } else if (message->enc == PROF_MSG_ENC_OMEMO) { ch = prefs_get_omemo_char(); } else { ch = strdup("-"); } win_insert_last_read_position_marker((ProfWin*)mucwin, mucwin->roomjid); wins_add_urls_ac(window, message, FALSE); wins_add_quotes_ac(window, message->plain, FALSE); if (g_slist_length(mentions) > 0) { _mucwin_print_mention(window, message->plain, message->from_jid->resourcepart, mynick, mentions, ch, flags); } else if (triggers) { win_print_them(window, THEME_ROOMTRIGGER, ch, flags, message->from_jid->resourcepart); _mucwin_print_triggers(window, message->plain, triggers); } else { win_println_incoming_muc_msg(window, ch, flags, message); } free(ch); } void mucwin_requires_config(ProfMucWin* mucwin) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; int num = wins_get_num(window); win_println(window, THEME_DEFAULT, "-", ""); win_println(window, THEME_ROOMINFO, "!", "Room locked, requires configuration."); win_println(window, THEME_ROOMINFO, "!", "Use '/room accept' to accept the defaults"); win_println(window, THEME_ROOMINFO, "!", "Use '/room destroy' to cancel and destroy the room"); win_println(window, THEME_ROOMINFO, "!", "Use '/room config' to edit the room configuration"); win_println(window, THEME_DEFAULT, "-", ""); // currently in groupchat window if (wins_is_current(window)) { status_bar_active(num, WIN_MUC, mucwin->roomjid); // not currently on groupchat window } else { status_bar_new(num, WIN_MUC, mucwin->roomjid); } } void mucwin_subject(ProfMucWin* mucwin, const char* const nick, const char* const subject) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; if (subject) { if (nick) { win_print(window, THEME_ROOMINFO, "!", "*%s has set the room subject: ", nick); win_appendln(window, THEME_DEFAULT, "%s", subject); } else { win_print(window, THEME_ROOMINFO, "!", "Room subject: "); win_appendln(window, THEME_DEFAULT, "%s", subject); } } else { if (nick) { win_println(window, THEME_ROOMINFO, "!", "*%s has cleared the room subject.", nick); } else { win_println(window, THEME_ROOMINFO, "!", "Room subject cleared"); } } } void mucwin_kick_error(ProfMucWin* mucwin, const char* const nick, const char* const error) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_println(window, THEME_ERROR, "!", "Error kicking %s: %s", nick, error); } void mucwin_broadcast(ProfMucWin* mucwin, const char* const message) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; int num = wins_get_num(window); win_print(window, THEME_ROOMINFO, "!", "Room message: "); win_appendln(window, THEME_DEFAULT, "%s", message); // currently in groupchat window if (wins_is_current(window)) { status_bar_active(num, WIN_MUC, mucwin->roomjid); // not currently on groupchat window } else { status_bar_new(num, WIN_MUC, mucwin->roomjid); } } void mucwin_affiliation_list_error(ProfMucWin* mucwin, const char* const affiliation, const char* const error) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_println(window, THEME_ERROR, "!", "Error retrieving %s list: %s", affiliation, error); } void mucwin_handle_affiliation_list(ProfMucWin* mucwin, const char* const affiliation, GSList* jids) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; if (jids) { win_println(window, THEME_DEFAULT, "!", "Affiliation: %s", affiliation); GSList* curr_jid = jids; while (curr_jid) { const char* jid = curr_jid->data; win_println(window, THEME_DEFAULT, "!", " %s", jid); curr_jid = g_slist_next(curr_jid); } win_println(window, THEME_DEFAULT, "!", ""); } else { win_println(window, THEME_DEFAULT, "!", "No users found with affiliation: %s", affiliation); win_println(window, THEME_DEFAULT, "!", ""); } } void mucwin_show_affiliation_list(ProfMucWin* mucwin, muc_affiliation_t affiliation) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; GSList* occupants = muc_occupants_by_affiliation(mucwin->roomjid, affiliation); if (!occupants) { switch (affiliation) { case MUC_AFFILIATION_OWNER: win_println(window, THEME_DEFAULT, "!", "No owners found."); break; case MUC_AFFILIATION_ADMIN: win_println(window, THEME_DEFAULT, "!", "No admins found."); break; case MUC_AFFILIATION_MEMBER: win_println(window, THEME_DEFAULT, "!", "No members found."); break; case MUC_AFFILIATION_OUTCAST: win_println(window, THEME_DEFAULT, "!", "No outcasts found."); break; case MUC_AFFILIATION_NONE: win_println(window, THEME_DEFAULT, "!", "No nones found."); break; default: break; } win_println(window, THEME_DEFAULT, "-", ""); } else { switch (affiliation) { case MUC_AFFILIATION_OWNER: win_println(window, THEME_DEFAULT, "!", "Owners:"); break; case MUC_AFFILIATION_ADMIN: win_println(window, THEME_DEFAULT, "!", "Admins:"); break; case MUC_AFFILIATION_MEMBER: win_println(window, THEME_DEFAULT, "!", "Members:"); break; case MUC_AFFILIATION_OUTCAST: win_println(window, THEME_DEFAULT, "!", "Outcasts:"); break; case MUC_AFFILIATION_NONE: win_println(window, THEME_DEFAULT, "!", "Nones:"); break; default: break; } GSList* curr_occupant = occupants; while (curr_occupant) { Occupant* occupant = curr_occupant->data; if (occupant->affiliation == affiliation) { if (occupant->jid) { win_println(window, THEME_DEFAULT, "!", " %s (%s)", occupant->nick, occupant->jid); } else { win_println(window, THEME_DEFAULT, "!", " %s", occupant->nick); } } curr_occupant = g_slist_next(curr_occupant); } win_println(window, THEME_DEFAULT, "-", ""); } } void mucwin_role_list_error(ProfMucWin* mucwin, const char* const role, const char* const error) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_println(window, THEME_ERROR, "!", "Error retrieving %s list: %s", role, error); } void mucwin_handle_role_list(ProfMucWin* mucwin, const char* const role, GSList* nicks) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; if (nicks) { win_println(window, THEME_DEFAULT, "!", "Role: %s", role); GSList* curr_nick = nicks; while (curr_nick) { const char* nick = curr_nick->data; Occupant* occupant = muc_roster_item(mucwin->roomjid, nick); if (occupant) { if (occupant->jid) { win_println(window, THEME_DEFAULT, "!", " %s (%s)", nick, occupant->jid); } else { win_println(window, THEME_DEFAULT, "!", " %s", nick); } } else { win_println(window, THEME_DEFAULT, "!", " %s", nick); } curr_nick = g_slist_next(curr_nick); } win_println(window, THEME_DEFAULT, "!", ""); } else { win_println(window, THEME_DEFAULT, "!", "No occupants found with role: %s", role); win_println(window, THEME_DEFAULT, "!", ""); } } void mucwin_show_role_list(ProfMucWin* mucwin, muc_role_t role) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; GSList* occupants = muc_occupants_by_role(mucwin->roomjid, role); if (!occupants) { switch (role) { case MUC_ROLE_MODERATOR: win_println(window, THEME_DEFAULT, "!", "No moderators found."); break; case MUC_ROLE_PARTICIPANT: win_println(window, THEME_DEFAULT, "!", "No participants found."); break; case MUC_ROLE_VISITOR: win_println(window, THEME_DEFAULT, "!", "No visitors found."); break; default: break; } win_println(window, THEME_DEFAULT, "-", ""); } else { switch (role) { case MUC_ROLE_MODERATOR: win_println(window, THEME_DEFAULT, "!", "Moderators:"); break; case MUC_ROLE_PARTICIPANT: win_println(window, THEME_DEFAULT, "!", "Participants:"); break; case MUC_ROLE_VISITOR: win_println(window, THEME_DEFAULT, "!", "Visitors:"); break; default: break; } GSList* curr_occupant = occupants; while (curr_occupant) { Occupant* occupant = curr_occupant->data; if (occupant->role == role) { if (occupant->jid) { win_println(window, THEME_DEFAULT, "!", " %s (%s)", occupant->nick, occupant->jid); } else { win_println(window, THEME_DEFAULT, "!", " %s", occupant->nick); } } curr_occupant = g_slist_next(curr_occupant); } win_println(window, THEME_DEFAULT, "-", ""); } } void mucwin_affiliation_set_error(ProfMucWin* mucwin, const char* const jid, const char* const affiliation, const char* const error) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_println(window, THEME_ERROR, "!", "Error setting %s affiliation for %s: %s", affiliation, jid, error); } void mucwin_role_set_error(ProfMucWin* mucwin, const char* const nick, const char* const role, const char* const error) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; win_println(window, THEME_ERROR, "!", "Error setting %s role for %s: %s", role, nick, error); } void mucwin_info(ProfMucWin* mucwin) { assert(mucwin != NULL); char* role = muc_role_str(mucwin->roomjid); char* affiliation = muc_affiliation_str(mucwin->roomjid); ProfWin* window = (ProfWin*)mucwin; win_println(window, THEME_DEFAULT, "!", "Room: %s", mucwin->roomjid); win_println(window, THEME_DEFAULT, "!", "Affiliation: %s", affiliation); win_println(window, THEME_DEFAULT, "!", "Role: %s", role); win_println(window, THEME_DEFAULT, "-", ""); } void mucwin_update_occupants(ProfMucWin* mucwin) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; if (win_has_active_subwin(window)) { occupantswin_occupants(mucwin->roomjid); } } void mucwin_show_occupants(ProfMucWin* mucwin) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; if (!win_has_active_subwin(window)) { wins_show_subwin(window); occupantswin_occupants(mucwin->roomjid); } } void mucwin_hide_occupants(ProfMucWin* mucwin) { assert(mucwin != NULL); ProfWin* window = (ProfWin*)mucwin; if (win_has_active_subwin(window)) { wins_hide_subwin(window); } } char* mucwin_get_string(ProfMucWin* mucwin) { assert(mucwin != NULL); if (mucwin->unread > 0) { return g_strdup_printf("Room %s, %d unread", mucwin->roomjid, mucwin->unread); } else { return g_strdup_printf("Room %s", mucwin->roomjid); } } void mucwin_set_enctext(ProfMucWin* mucwin, const char* const enctext) { if (mucwin->enctext) { free(mucwin->enctext); } mucwin->enctext = strdup(enctext); } void mucwin_unset_enctext(ProfMucWin* mucwin) { if (mucwin->enctext) { free(mucwin->enctext); mucwin->enctext = NULL; } } void mucwin_set_message_char(ProfMucWin* mucwin, const char* const ch) { if (mucwin->message_char) { free(mucwin->message_char); } mucwin->message_char = strdup(ch); } void mucwin_unset_message_char(ProfMucWin* mucwin) { if (mucwin->message_char) { free(mucwin->message_char); mucwin->message_char = NULL; } } static void _mucwin_set_last_message(ProfMucWin* mucwin, const char* const id, const char* const message) { free(mucwin->last_message); mucwin->last_message = strdup(message); free(mucwin->last_msg_id); mucwin->last_msg_id = strdup(id); }