/** * @file llparticipantlist.cpp * @brief LLParticipantList : model of a conversation session with added speaker events handling * * $LicenseInfo:firstyear=2009&license=viewerlgpl$ * Second Life Viewer Source Code * Copyright (C) 2010, Linden Research, Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; * version 2.1 of the License only. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * * Linden Research, Inc., 945 Battery Street, San Francisco, CA 94111 USA * $/LicenseInfo$ */ #include "llviewerprecompiledheaders.h" #include "llavatarnamecache.h" #include "llerror.h" #include "llimview.h" #include "llfloaterimcontainer.h" #include "llparticipantlist.h" #include "llspeakers.h" //LLParticipantList retrieves add, clear and remove events and updates view accordingly #if LL_MSVC #pragma warning (disable : 4355) // 'this' used in initializer list: yes, intentionally #endif // See EXT-4301. /** * class LLAvalineUpdater - observe the list of voice participants in session and check * presence of Avaline Callers among them. * * LLAvalineUpdater is a LLVoiceClientParticipantObserver. It provides two kinds of validation: * - whether Avaline caller presence among participants; * - whether watched Avaline caller still exists in voice channel. * Both validations have callbacks which will notify subscriber if any of event occur. * * @see findAvalineCaller() * @see checkIfAvalineCallersExist() */ class LLAvalineUpdater : public LLVoiceClientParticipantObserver { public: typedef boost::function process_avaline_callback_t; LLAvalineUpdater(process_avaline_callback_t found_cb, process_avaline_callback_t removed_cb) : mAvalineFoundCallback(found_cb) , mAvalineRemovedCallback(removed_cb) { LLVoiceClient::getInstance()->addObserver(this); } ~LLAvalineUpdater() { if (LLVoiceClient::instanceExists()) { LLVoiceClient::getInstance()->removeObserver(this); } } /** * Adds UUID of Avaline caller to watch. * * @see checkIfAvalineCallersExist(). */ void watchAvalineCaller(const LLUUID& avaline_caller_id) { mAvalineCallers.insert(avaline_caller_id); } void onParticipantsChanged() { uuid_set_t participant_uuids; LLVoiceClient::getInstance()->getParticipantList(participant_uuids); // check whether Avaline caller exists among voice participants // and notify Participant List findAvalineCaller(participant_uuids); // check whether watched Avaline callers still present among voice participant // and remove if absents. checkIfAvalineCallersExist(participant_uuids); } private: typedef std::set uuid_set_t; /** * Finds Avaline callers among voice participants and calls mAvalineFoundCallback. * * When Avatar is in group call with Avaline caller and then ends call Avaline caller stays * in Group Chat floater (exists in LLSpeakerMgr). If Avatar starts call with that group again * Avaline caller is added to voice channel AFTER Avatar is connected to group call. * But Voice Control Panel (VCP) is filled from session LLSpeakerMgr and there is no information * if a speaker is Avaline caller. * * In this case this speaker is created as avatar and will be recreated when it appears in * Avatar's Voice session. * * @see LLParticipantList::onAvalineCallerFound() */ void findAvalineCaller(const uuid_set_t& participant_uuids) { uuid_set_t::const_iterator it = participant_uuids.begin(), it_end = participant_uuids.end(); for(; it != it_end; ++it) { const LLUUID& participant_id = *it; if (!LLVoiceClient::getInstance()->isParticipantAvatar(participant_id)) { LL_DEBUGS("Avaline") << "Avaline caller found among voice participants: " << participant_id << LL_ENDL; if (mAvalineFoundCallback) { mAvalineFoundCallback(participant_id); } } } } /** * Finds Avaline callers which are not anymore among voice participants and calls mAvalineRemovedCallback. * * The problem is when Avaline caller ends a call it is removed from Voice Client session but * still exists in LLSpeakerMgr. Server does not send such information. * This method implements a HUCK to notify subscribers that watched Avaline callers by class * are not anymore in the call. * * @see LLParticipantList::onAvalineCallerRemoved() */ void checkIfAvalineCallersExist(const uuid_set_t& participant_uuids) { uuid_set_t::iterator it = mAvalineCallers.begin(); uuid_set_t::const_iterator participants_it_end = participant_uuids.end(); while (it != mAvalineCallers.end()) { const LLUUID participant_id = *it; LL_DEBUGS("Avaline") << "Check avaline caller: " << participant_id << LL_ENDL; bool not_found = participant_uuids.find(participant_id) == participants_it_end; if (not_found) { LL_DEBUGS("Avaline") << "Watched Avaline caller is not found among voice participants: " << participant_id << LL_ENDL; // notify Participant List if (mAvalineRemovedCallback) { mAvalineRemovedCallback(participant_id); } // remove from the watch list mAvalineCallers.erase(it++); } else { ++it; } } } process_avaline_callback_t mAvalineFoundCallback; process_avaline_callback_t mAvalineRemovedCallback; uuid_set_t mAvalineCallers; }; LLParticipantList::LLParticipantList(LLSpeakerMgr* data_source, LLFolderViewModelInterface& root_view_model) : LLConversationItemSession(data_source->getSessionID(), root_view_model), mSpeakerMgr(data_source), mValidateSpeakerCallback(NULL) { mAvalineUpdater = new LLAvalineUpdater(boost::bind(&LLParticipantList::onAvalineCallerFound, this, _1), boost::bind(&LLParticipantList::onAvalineCallerRemoved, this, _1)); mSpeakerAddListener = new SpeakerAddListener(*this); mSpeakerRemoveListener = new SpeakerRemoveListener(*this); mSpeakerClearListener = new SpeakerClearListener(*this); mSpeakerModeratorListener = new SpeakerModeratorUpdateListener(*this); mSpeakerUpdateListener = new SpeakerUpdateListener(*this); mSpeakerMuteListener = new SpeakerMuteListener(*this); mSpeakerMgr->addListener(mSpeakerAddListener, "add"); mSpeakerMgr->addListener(mSpeakerRemoveListener, "remove"); mSpeakerMgr->addListener(mSpeakerClearListener, "clear"); mSpeakerMgr->addListener(mSpeakerModeratorListener, "update_moderator"); mSpeakerMgr->addListener(mSpeakerUpdateListener, "update_speaker"); setSessionID(mSpeakerMgr->getSessionID()); //Lets fill avatarList with existing speakers LLSpeakerMgr::speaker_list_t speaker_list; mSpeakerMgr->getSpeakerList(&speaker_list, true); for(LLSpeakerMgr::speaker_list_t::iterator it = speaker_list.begin(); it != speaker_list.end(); it++) { const LLPointer& speakerp = *it; addAvatarIDExceptAgent(speakerp->mID); if ( speakerp->mIsModerator ) { mModeratorList.insert(speakerp->mID); } else { mModeratorToRemoveList.insert(speakerp->mID); } } // Identify and store what kind of session we are LLIMModel::LLIMSession* im_session = LLIMModel::getInstance()->findIMSession(data_source->getSessionID()); if (im_session) { // By default, sessions that can't be identified as group or ad-hoc will be considered P2P (i.e. 1 on 1) mConvType = CONV_SESSION_1_ON_1; if (im_session->isAdHocSessionType()) { mConvType = CONV_SESSION_AD_HOC; } else if (im_session->isGroupSessionType()) { mConvType = CONV_SESSION_GROUP; } } else { // That's the only session that doesn't get listed in the LLIMModel as a session... mConvType = CONV_SESSION_NEARBY; } } LLParticipantList::~LLParticipantList() { delete mAvalineUpdater; } /* Seems this method is not necessary after onAvalineCallerRemoved was implemented; It does nothing because list item is always created with correct class type for Avaline caller. For now Avaline Caller is removed from the LLSpeakerMgr List when it is removed from the Voice Client session. This happens in two cases: if Avaline Caller ends call itself or if Resident ends group call. Probably Avaline caller should be removed from the LLSpeakerMgr list ONLY if it ends call itself. Asked in EXT-4301. */ void LLParticipantList::onAvalineCallerFound(const LLUUID& participant_id) { removeParticipant(participant_id); // re-add avaline caller with a correct class instance. addAvatarIDExceptAgent(participant_id); } void LLParticipantList::onAvalineCallerRemoved(const LLUUID& participant_id) { LL_DEBUGS("Avaline") << "Removing avaline caller from the list: " << participant_id << LL_ENDL; mSpeakerMgr->removeAvalineSpeaker(participant_id); } void LLParticipantList::setValidateSpeakerCallback(validate_speaker_callback_t cb) { mValidateSpeakerCallback = cb; } void LLParticipantList::update() { mSpeakerMgr->update(true); } bool LLParticipantList::onAddItemEvent(LLPointer event, const LLSD& userdata) { LLUUID uu_id = event->getValue().asUUID(); if (mValidateSpeakerCallback && !mValidateSpeakerCallback(uu_id)) { return true; } addAvatarIDExceptAgent(uu_id); return true; } bool LLParticipantList::onRemoveItemEvent(LLPointer event, const LLSD& userdata) { LLUUID avatar_id = event->getValue().asUUID(); removeParticipant(avatar_id); return true; } bool LLParticipantList::onClearListEvent(LLPointer event, const LLSD& userdata) { clearParticipants(); return true; } bool LLParticipantList::onSpeakerUpdateEvent(LLPointer event, const LLSD& userdata) { const LLSD& evt_data = event->getValue(); if ( evt_data.has("id") ) { LLUUID participant_id = evt_data["id"]; LLFloaterIMContainer* im_box = LLFloaterIMContainer::findInstance(); if (im_box) { im_box->setTimeNow(mUUID,participant_id); } } return true; } bool LLParticipantList::onModeratorUpdateEvent(LLPointer event, const LLSD& userdata) { const LLSD& evt_data = event->getValue(); if ( evt_data.has("id") && evt_data.has("is_moderator") ) { LLUUID id = evt_data["id"]; bool is_moderator = evt_data["is_moderator"]; if ( id.notNull() ) { if ( is_moderator ) mModeratorList.insert(id); else { std::set::iterator it = mModeratorList.find (id); if ( it != mModeratorList.end () ) { mModeratorToRemoveList.insert(id); mModeratorList.erase(id); } } // *TODO : do we have to fire an event so that LLFloaterIMSessionTab::refreshConversation() gets called } } return true; } bool LLParticipantList::onSpeakerMuteEvent(LLPointer event, const LLSD& userdata) { LLPointer speakerp = (LLSpeaker*)event->getSource(); if (speakerp.isNull()) return false; // update UI on confirmation of moderator mutes if (event->getValue().asString() == "voice") { setParticipantIsMuted(speakerp->mID, speakerp->mModeratorMutedVoice); } return true; } void LLParticipantList::addAvatarIDExceptAgent(const LLUUID& avatar_id) { // Do not add if already in there, is the session id (hence not an avatar) or excluded for some reason if (findParticipant(avatar_id) || (avatar_id == mUUID)) { return; } bool is_avatar = LLVoiceClient::getInstance()->isParticipantAvatar(avatar_id); LLConversationItemParticipant* participant = NULL; if (is_avatar) { // Create a participant view model instance LLAvatarName avatar_name; bool has_name = LLAvatarNameCache::get(avatar_id, &avatar_name); participant = new LLConversationItemParticipant(!has_name ? LLTrans::getString("AvatarNameWaiting") : avatar_name.getDisplayName() , avatar_id, mRootViewModel); participant->fetchAvatarName(); } else { std::string display_name = LLVoiceClient::getInstance()->getDisplayName(avatar_id); // Create a participant view model instance participant = new LLConversationItemParticipant(display_name.empty() ? LLTrans::getString("AvatarNameWaiting") : display_name, avatar_id, mRootViewModel); mAvalineUpdater->watchAvalineCaller(avatar_id); } // *TODO : Need to update the online/offline status of the participant // Hack for this: LLAvatarTracker::instance().isBuddyOnline(avatar_id)) // Add the participant model to the session's children list // This will post "add_participant" event addParticipant(participant); adjustParticipant(avatar_id); } static LLFastTimer::DeclareTimer FTM_FOLDERVIEW_TEST("add test avatar agents"); void LLParticipantList::adjustParticipant(const LLUUID& speaker_id) { LLPointer speakerp = mSpeakerMgr->findSpeaker(speaker_id); if (speakerp.isNull()) return; // add listener to process moderation changes speakerp->addListener(mSpeakerMuteListener); } // // LLParticipantList::SpeakerAddListener // bool LLParticipantList::SpeakerAddListener::handleEvent(LLPointer event, const LLSD& userdata) { /** * We need to filter speaking objects. These objects shouldn't appear in the list. * @see LLFloaterChat::addChat() in llviewermessage.cpp to get detailed call hierarchy */ const LLUUID& speaker_id = event->getValue().asUUID(); LLPointer speaker = mParent.mSpeakerMgr->findSpeaker(speaker_id); if (speaker.isNull() || (speaker->mType == LLSpeaker::SPEAKER_OBJECT)) { return false; } return mParent.onAddItemEvent(event, userdata); } // // LLParticipantList::SpeakerRemoveListener // bool LLParticipantList::SpeakerRemoveListener::handleEvent(LLPointer event, const LLSD& userdata) { return mParent.onRemoveItemEvent(event, userdata); } // // LLParticipantList::SpeakerClearListener // bool LLParticipantList::SpeakerClearListener::handleEvent(LLPointer event, const LLSD& userdata) { return mParent.onClearListEvent(event, userdata); } // // LLParticipantList::SpeakerUpdateListener // bool LLParticipantList::SpeakerUpdateListener::handleEvent(LLPointer event, const LLSD& userdata) { return mParent.onSpeakerUpdateEvent(event, userdata); } // // LLParticipantList::SpeakerModeratorListener // bool LLParticipantList::SpeakerModeratorUpdateListener::handleEvent(LLPointer event, const LLSD& userdata) { return mParent.onModeratorUpdateEvent(event, userdata); } bool LLParticipantList::SpeakerMuteListener::handleEvent(LLPointer event, const LLSD& userdata) { return mParent.onSpeakerMuteEvent(event, userdata); } //EOF