/** * @file llavataractions.cpp * @brief Friend-related actions (add, remove, offer teleport, etc) * * $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 "llavataractions.h" #include "boost/lambda/lambda.hpp" // for lambda::constant #include "llavatarnamecache.h" // IDEVO #include "llsd.h" #include "llnotifications.h" #include "llnotificationsutil.h" #include "roles_constants.h" // for GP_MEMBER_INVITE #include "llagent.h" #include "llappviewer.h" // for gLastVersionChannel #include "llcachename.h" #include "llcallingcard.h" // for LLAvatarTracker #include "llconversationlog.h" #include "llfloateravatarpicker.h" // for LLFloaterAvatarPicker #include "llfloaterconversationpreview.h" #include "llfloatergroupinvite.h" #include "llfloatergroups.h" #include "llfloaterreg.h" #include "llfloaterpay.h" #include "llfloaterprofile.h" #include "llfloatersidepanelcontainer.h" #include "llfloaterwebcontent.h" #include "llfloaterworldmap.h" #include "llfolderview.h" #include "llgiveinventory.h" #include "llinventorybridge.h" #include "llinventorymodel.h" // for gInventory.findCategoryUUIDForType #include "llinventorypanel.h" #include "llfloaterimcontainer.h" #include "llimview.h" // for gIMMgr #include "llmutelist.h" #include "llnotificationsutil.h" // for LLNotificationsUtil #include "llpaneloutfitedit.h" #include "llpanelprofile.h" #include "llparcel.h" #include "llrecentpeople.h" #include "lltrans.h" #include "llviewercontrol.h" #include "llviewerobjectlist.h" #include "llviewermessage.h" // for handle_lure #include "llviewernetwork.h" //LLGridManager #include "llviewerparcelmgr.h" #include "llviewerregion.h" #include "lltrans.h" #include "llcallingcard.h" #include "llslurl.h" // IDEVO #include "llsidepanelinventory.h" #include "llavatarname.h" #include "llagentui.h" #include "lluiusage.h" // [RLVa:KB] - Checked: 2011-04-11 (RLVa-1.3.0) #include "rlvactions.h" #include "rlvcommon.h" #include "rlvhandler.h" // [/RLVa:KB] // Firestorm includes #include "fsfloaterim.h" #include "fsfloaterimcontainer.h" #include "fslslbridge.h" #include "fsradar.h" #include "fsassetblacklist.h" #include "llagentpicksinfo.h" #include "llfloaterregioninfo.h" #include "llfloaterreporter.h" #include "llparcel.h" #include "lltrans.h" #include "llviewermenu.h" #include "llviewernetwork.h" #include "llviewerparcelmgr.h" #include "llviewerwindow.h" // For opening logs externally #include "llworld.h" // Flags for kick message const U32 KICK_FLAGS_DEFAULT = 0x0; const U32 KICK_FLAGS_FREEZE = 1 << 0; const U32 KICK_FLAGS_UNFREEZE = 1 << 1; std::string getProfileURL(const std::string& agent_name, bool feed_only) { // OpenSim support //std::string url = "[WEB_PROFILE_URL][AGENT_NAME][FEED_ONLY]"; //LLSD subs; //subs["WEB_PROFILE_URL"] = LLGridManager::getInstance()->getWebProfileURL(); //subs["AGENT_NAME"] = agent_name; //subs["FEED_ONLY"] = feed_only ? "/?feed_only=true" : ""; std::string url; LLSD subs; #ifdef OPENSIM if (LLGridManager::instance().isInOpenSim()) { url = LLGridManager::getInstance()->getWebProfileURL(); if (url.empty()) { return LLStringUtil::null; } std::string match = "?name=[AGENT_NAME]"; if (url.find(match) == std::string::npos) { url += match; } } else #endif { url = "[WEB_PROFILE_URL][AGENT_NAME][FEED_ONLY]"; subs["FEED_ONLY"] = feed_only ? "/?feed_only=true" : ""; subs["WEB_PROFILE_URL"] = LLGridManager::getInstance()->getWebProfileURL(); } // subs["AGENT_NAME"] = agent_name; url = LLWeb::expandURLSubstitutions(url, subs); LLStringUtil::toLower(url); return url; } // static void LLAvatarActions::requestFriendshipDialog(const LLUUID& id, const std::string& name) { if(id == gAgentID) { LLNotificationsUtil::add("AddSelfFriend"); return; } LLSD args; args["NAME"] = LLSLURL("agent", id, "completename").getSLURLString(); LLSD payload; payload["id"] = id; payload["name"] = name; LLNotificationsUtil::add("AddFriendWithMessage", args, payload, &callbackAddFriendWithMessage); // add friend to recent people list LLRecentPeople::instance().add(id); } static void on_avatar_name_friendship(const LLUUID& id, const LLAvatarName av_name) { LLAvatarActions::requestFriendshipDialog(id, av_name.getCompleteName()); } // static void LLAvatarActions::requestFriendshipDialog(const LLUUID& id) { if(id.isNull()) { return; } LLAvatarNameCache::get(id, boost::bind(&on_avatar_name_friendship, _1, _2)); } // static void LLAvatarActions::removeFriendDialog(const LLUUID& id) { if (id.isNull()) return; uuid_vec_t ids; ids.push_back(id); removeFriendsDialog(ids); } // static void LLAvatarActions::removeFriendsDialog(const uuid_vec_t& ids) { if(ids.size() == 0) return; LLSD args; std::string msgType; if(ids.size() == 1) { LLUUID agent_id = ids[0]; LLAvatarName av_name; if(LLAvatarNameCache::get(agent_id, &av_name)) { args["NAME"] = av_name.getCompleteName(); } msgType = "RemoveFromFriends"; } else { msgType = "RemoveMultipleFromFriends"; } LLSD payload; for (uuid_vec_t::const_iterator it = ids.begin(); it != ids.end(); ++it) { payload["ids"].append(*it); } LLNotificationsUtil::add(msgType, args, payload, &handleRemove); } // static void LLAvatarActions::offerTeleport(const LLUUID& invitee) { if (invitee.isNull()) return; std::vector ids; ids.push_back(invitee); offerTeleport(ids); } // static void LLAvatarActions::offerTeleport(const uuid_vec_t& ids) { if (ids.size() == 0) return; // Fix edge case with multi-selection containing offlines //handle_lure(ids); uuid_vec_t result; for (uuid_vec_t::const_iterator it = ids.begin(); it != ids.end(); ++it) { if (canOfferTeleport(*it)) { result.push_back(*it); // Maximum 250 lures if (result.size() == 250) { break; } } } handle_lure(result); // } static void on_avatar_name_cache_start_im(const LLUUID& agent_id, const LLAvatarName& av_name) { std::string name = av_name.getDisplayName(); LLUUID session_id = gIMMgr->addSession(name, IM_NOTHING_SPECIAL, agent_id); if (session_id != LLUUID::null) { // [FS communication UI] //LLFloaterIMContainer::getInstance()->showConversation(session_id); FSFloaterIM::show(session_id); // [FS communication UI] } make_ui_sound("UISndStartIM"); } // static void LLAvatarActions::startIM(const LLUUID& id) { if (id.isNull() || gAgent.getID() == id) return; // [RLVa:KB] - Checked: 2013-05-09 (RLVa-1.4.9) if (!RlvActions::canStartIM(id)) { make_ui_sound("UISndInvalidOp"); RlvUtil::notifyBlocked(RlvStringKeys::Blocked::StartIm, LLSD().with("RECIPIENT", LLSLURL("agent", id, "completename").getSLURLString())); return; } // [/RLVa:KB] LLAvatarNameCache::get(id, boost::bind(&on_avatar_name_cache_start_im, _1, _2)); } // static void LLAvatarActions::endIM(const LLUUID& id) { if (id.isNull()) return; LLUUID session_id = gIMMgr->computeSessionID(IM_NOTHING_SPECIAL, id); if (session_id != LLUUID::null) { gIMMgr->leaveSession(session_id); } } static void on_avatar_name_cache_start_call(const LLUUID& agent_id, const LLAvatarName& av_name) { std::string name = av_name.getDisplayName(); LLUUID session_id = gIMMgr->addSession(name, IM_NOTHING_SPECIAL, agent_id, LLSD()); gIMMgr->autoStartCallOnStartup(session_id); make_ui_sound("UISndStartIM"); } // static void LLAvatarActions::startCall(const LLUUID& id) { if (id.isNull()) { return; } // [RLVa:KB] - Checked: 2013-05-09 (RLVa-1.4.9) if (!RlvActions::canStartIM(id)) { make_ui_sound("UISndInvalidOp"); RlvUtil::notifyBlocked(RlvStringKeys::Blocked::StartIm, LLSD().with("RECIPIENT", LLSLURL("agent", id, "completename").getSLURLString())); return; } // [/RLVa:KB] LLAvatarNameCache::get(id, boost::bind(&on_avatar_name_cache_start_call, _1, _2)); } // static // [FS Communication UI] //void LLAvatarActions::startAdhocCall(const uuid_vec_t& ids, const LLUUID& floater_id) const LLUUID LLAvatarActions::startAdhocCall(const uuid_vec_t& ids, const LLUUID& floater_id) // { if (ids.size() == 0) { // [FS Communication UI] //return; return LLUUID::null; // } // convert vector into std::vector for addSession std::vector id_array; id_array.reserve(ids.size()); for (uuid_vec_t::const_iterator it = ids.begin(); it != ids.end(); ++it) { // [RLVa:KB] - Checked: 2011-04-11 (RLVa-1.3.0) const LLUUID& idAgent = *it; if (!RlvActions::canStartIM(idAgent)) { make_ui_sound("UISndInvalidOp"); RlvUtil::notifyBlocked(RlvStringKeys::Blocked::StartConference); return LLUUID::null; } id_array.push_back(idAgent); // [/RLVa:KB] // id_array.push_back(*it); } // create the new ad hoc voice session const std::string title = LLTrans::getString("conference-title"); LLUUID session_id = gIMMgr->addSession(title, IM_SESSION_CONFERENCE_START, ids[0], id_array, LLSD(), floater_id); if (session_id == LLUUID::null) { // [FS Communication UI] //return; return session_id; // } gIMMgr->autoStartCallOnStartup(session_id); make_ui_sound("UISndStartIM"); // [FS Communication UI] return session_id; } /* AD *TODO: Is this function needed any more? I fixed it a bit(added check for canCall), but it appears that it is not used anywhere. Maybe it should be removed? // static bool LLAvatarActions::isCalling(const LLUUID &id) { if (id.isNull() || !canCall()) { return false; } LLUUID session_id = gIMMgr->computeSessionID(IM_NOTHING_SPECIAL, id); return (LLIMModel::getInstance()->findIMSession(session_id) != NULL); }*/ //static bool LLAvatarActions::canCall() { return LLVoiceClient::getInstance()->voiceEnabled() && LLVoiceClient::getInstance()->isVoiceWorking(); } // static // [FS Communication UI] //void LLAvatarActions::startConference(const uuid_vec_t& ids, const LLUUID& floater_id) const LLUUID LLAvatarActions::startConference(const uuid_vec_t& ids, const LLUUID& floater_id) // { // *HACK: Copy into dynamic array std::vector id_array; id_array.reserve(ids.size()); for (uuid_vec_t::const_iterator it = ids.begin(); it != ids.end(); ++it) { // [RLVa:KB] - Checked: 2011-04-11 (RLVa-1.3.0) const LLUUID& idAgent = *it; if (!RlvActions::canStartIM(idAgent)) { make_ui_sound("UISndInvalidOp"); RlvUtil::notifyBlocked(RlvStringKeys::Blocked::StartConference); return LLUUID::null; } id_array.push_back(idAgent); // [/RLVa:KB] // id_array.push_back(*it); } const std::string title = LLTrans::getString("conference-title"); LLUUID session_id = gIMMgr->addSession(title, IM_SESSION_CONFERENCE_START, ids[0], id_array, LLSD(), floater_id); if (session_id == LLUUID::null) { // [FS Communication UI] //return; return session_id; // } // [FS communication UI] //FSFloaterIMContainer::getInstance()->showConversation(session_id); FSFloaterIM::show(session_id); // [FS communication UI] make_ui_sound("UISndStartIM"); // [FS Communication UI] return session_id; } // static void LLAvatarActions::showProfile(const LLUUID& avatar_id) { if (avatar_id.notNull()) { LLFloaterReg::showInstance("profile", LLSD().with("id", avatar_id)); } } // static void LLAvatarActions::showPicks(const LLUUID& avatar_id) { if (avatar_id.notNull()) { LLFloaterProfile* profilefloater = dynamic_cast(LLFloaterReg::showInstance("profile", LLSD().with("id", avatar_id))); if (profilefloater) { profilefloater->showPick(); } } } // static void LLAvatarActions::showPick(const LLUUID& avatar_id, const LLUUID& pick_id) { if (avatar_id.notNull()) { LLFloaterProfile* profilefloater = dynamic_cast(LLFloaterReg::showInstance("profile", LLSD().with("id", avatar_id))); if (profilefloater) { profilefloater->showPick(pick_id); } } } // static void LLAvatarActions::createPick() { // FIRE-7694 / BUG-932 / MAINT-1999 if (LLAgentPicksInfo::getInstance()->isPickLimitReached()) { LLNotificationsUtil::add("PickLimitReached"); return; } // LLFloaterProfile* profilefloater = dynamic_cast(LLFloaterReg::showInstance("profile", LLSD().with("id", gAgent.getID()))); LLViewerRegion* region = gAgent.getRegion(); if (profilefloater && region) { LLPickData data; data.pos_global = gAgent.getPositionGlobal(); data.sim_name = region->getName(); LLParcel* parcel = LLViewerParcelMgr::getInstance()->getAgentParcel(); if (parcel) { data.name = parcel->getName(); data.desc = parcel->getDesc(); data.snapshot_id = parcel->getSnapshotID(); data.parcel_id = parcel->getID(); } else { data.name = region->getName(); } profilefloater->createPick(data); } } // static bool LLAvatarActions::isPickTabSelected(const LLUUID& avatar_id) { if (avatar_id.notNull()) { LLFloaterProfile* profilefloater = LLFloaterReg::findTypedInstance("profile", LLSD().with("id", avatar_id)); if (profilefloater) { return profilefloater->isPickTabSelected(); } } return false; } // static void LLAvatarActions::showClassifieds(const LLUUID& avatar_id) { if (avatar_id.notNull()) { LLFloaterProfile* profilefloater = dynamic_cast(LLFloaterReg::showInstance("profile", LLSD().with("id", avatar_id))); if (profilefloater) { profilefloater->showClassified(); } } } // static void LLAvatarActions::showClassified(const LLUUID& avatar_id, const LLUUID& classified_id, bool edit) { if (avatar_id.notNull()) { LLFloaterProfile* profilefloater = dynamic_cast(LLFloaterReg::showInstance("profile", LLSD().with("id", avatar_id))); if (profilefloater) { profilefloater->showClassified(classified_id, edit); } } } // static void LLAvatarActions::createClassified() { LLFloaterProfile* profilefloater = dynamic_cast(LLFloaterReg::showInstance("profile", LLSD().with("id", gAgent.getID()))); if (profilefloater) { profilefloater->createClassified(); } } //static bool LLAvatarActions::profileVisible(const LLUUID& avatar_id) { LLSD sd; sd["id"] = avatar_id; LLFloater* floater = getProfileFloater(avatar_id); return floater && floater->isShown(); } //static LLFloater* LLAvatarActions::getProfileFloater(const LLUUID& avatar_id) { LLFloaterProfile* floater = LLFloaterReg::findTypedInstance("profile", LLSD().with("id", avatar_id)); return floater; } //static void LLAvatarActions::hideProfile(const LLUUID& avatar_id) { LLSD sd; sd["id"] = avatar_id; LLFloater* floater = getProfileFloater(avatar_id); if (floater) { floater->closeFloater(); } } // static void LLAvatarActions::showOnMap(const LLUUID& id) { LLAvatarName av_name; if (!LLAvatarNameCache::get(id, &av_name)) { LLAvatarNameCache::get(id, boost::bind(&LLAvatarActions::showOnMap, id)); return; } gFloaterWorldMap->trackAvatar(id, av_name.getDisplayName()); LLFloaterReg::showInstance("world_map", "center"); } // static void LLAvatarActions::pay(const LLUUID& id) { LLNotification::Params params("DoNotDisturbModePay"); params.functor.function(boost::bind(&LLAvatarActions::handlePay, _1, _2, id)); if (gAgent.isDoNotDisturb()) { // warn users of being in do not disturb mode during a transaction LLNotifications::instance().add(params); } else { LLNotifications::instance().forceResponse(params, 1); } } void LLAvatarActions::teleport_request_callback(const LLSD& notification, const LLSD& response) { S32 option; if (response.isInteger()) { option = response.asInteger(); } else { option = LLNotificationsUtil::getSelectedOption(notification, response); } if (0 == option) { LLMessageSystem* msg = gMessageSystem; // [RLVa:KB] - Checked: RLVa-2.0.0 const LLUUID idRecipient = notification["substitutions"]["uuid"]; std::string strMessage = response["message"]; // Filter the request message if the recipients is IM-blocked if ( (RlvActions::isRlvEnabled()) && ((!RlvActions::canStartIM(idRecipient)) || (!RlvActions::canSendIM(idRecipient))) ) { strMessage = RlvStrings::getString(RlvStringKeys::Hidden::Generic); } // [/RLVa:KB] msg->newMessageFast(_PREHASH_ImprovedInstantMessage); msg->nextBlockFast(_PREHASH_AgentData); msg->addUUIDFast(_PREHASH_AgentID, gAgent.getID()); msg->addUUIDFast(_PREHASH_SessionID, gAgent.getSessionID()); msg->nextBlockFast(_PREHASH_MessageBlock); msg->addBOOLFast(_PREHASH_FromGroup, false); msg->addUUIDFast(_PREHASH_ToAgentID, notification["substitutions"]["uuid"] ); msg->addU8Fast(_PREHASH_Offline, IM_ONLINE); msg->addU8Fast(_PREHASH_Dialog, IM_TELEPORT_REQUEST); msg->addUUIDFast(_PREHASH_ID, LLUUID::null); msg->addU32Fast(_PREHASH_Timestamp, NO_TIMESTAMP); // no timestamp necessary std::string name; LLAgentUI::buildFullname(name); msg->addStringFast(_PREHASH_FromAgentName, name); // [RLVa:KB] - Checked: RLVa-2.0.0 msg->addStringFast(_PREHASH_Message, strMessage); // [/RLVa:KB] // msg->addStringFast(_PREHASH_Message, response["message"]); msg->addU32Fast(_PREHASH_ParentEstateID, 0); msg->addUUIDFast(_PREHASH_RegionID, LLUUID::null); msg->addVector3Fast(_PREHASH_Position, gAgent.getPositionAgent()); gMessageSystem->addBinaryDataFast( _PREHASH_BinaryBucket, EMPTY_BINARY_BUCKET, EMPTY_BINARY_BUCKET_SIZE); gAgent.sendReliableMessage(); } } // static void LLAvatarActions::teleportRequest(const LLUUID& id) { // Warn if 'reject teleport offers and requests' mode is active if (gSavedPerAccountSettings.getBOOL("FSRejectTeleportOffersMode")) { LLNotificationsUtil::add("RejectTeleportOffersModeWarning"); return; } // LLSD notification; notification["uuid"] = id; // [RLVa:KB] - Checked: RLVa-1.5.0 notification["NAME"] = LLSLURL("agent", id, (RlvActions::canShowName(RlvActions::SNC_TELEPORTREQUEST, id)) ? "completename" : "rlvanonym").getSLURLString(); // [/RLVa:KB] // LLAvatarName av_name; // if (!LLAvatarNameCache::get(id, &av_name)) // { // // unlikely ... they just picked this name from somewhere... // LLAvatarNameCache::get(id, boost::bind(&LLAvatarActions::teleportRequest, id)); // return; // reinvoke this when the name resolves // } // notification["NAME"] = av_name.getCompleteName(); LLSD payload; LLNotificationsUtil::add("TeleportRequestPrompt", notification, payload, teleport_request_callback); } // static void LLAvatarActions::kick(const LLUUID& id) { LLSD payload; payload["avatar_id"] = id; LLNotifications::instance().add("KickUser", LLSD(), payload, handleKick); } // static void LLAvatarActions::freezeAvatar(const LLUUID& id) { LLAvatarName av_name; LLSD payload; payload["avatar_id"] = id; if (LLAvatarNameCache::get(id, &av_name)) { LLSD args; args["AVATAR_NAME"] = av_name.getUserName(); LLNotificationsUtil::add("FreezeAvatarFullname", args, payload, handleFreezeAvatar); } else { LLNotificationsUtil::add("FreezeAvatar", LLSD(), payload, handleFreezeAvatar); } } // static void LLAvatarActions::ejectAvatar(const LLUUID& id, bool ban_enabled) { LLAvatarName av_name; LLSD payload; payload["avatar_id"] = id; payload["ban_enabled"] = ban_enabled; LLSD args; bool has_name = LLAvatarNameCache::get(id, &av_name); if (has_name) { args["AVATAR_NAME"] = av_name.getUserName(); } if (ban_enabled) { LLNotificationsUtil::add("EjectAvatarFullname", args, payload, handleEjectAvatar); } else { if (has_name) { LLNotificationsUtil::add("EjectAvatarFullnameNoBan", args, payload, handleEjectAvatar); } else { LLNotificationsUtil::add("EjectAvatarNoBan", LLSD(), payload, handleEjectAvatar); } } } // static void LLAvatarActions::freeze(const LLUUID& id) { LLSD payload; payload["avatar_id"] = id; LLNotifications::instance().add("FreezeUser", LLSD(), payload, handleFreeze); } // static void LLAvatarActions::unfreeze(const LLUUID& id) { LLSD payload; payload["avatar_id"] = id; LLNotifications::instance().add("UnFreezeUser", LLSD(), payload, handleUnfreeze); } //static void LLAvatarActions::csr(const LLUUID& id, std::string name) { if (name.empty()) return; std::string url = "http://csr.lindenlab.com/agent/"; // slow and stupid, but it's late auto len = name.length(); for (size_t i = 0; i < len; i++) { if (name[i] == ' ') { url += "%20"; } else { url += name[i]; } } LLWeb::loadURL(url); } //static void LLAvatarActions::share(const LLUUID& id) { // [RLVa:KB] - @share if ( (RlvActions::isRlvEnabled()) && (!RlvActions::canGiveInventory(id)) ) { RlvUtil::notifyBlocked(RlvStringKeys::Blocked::Share, LLSD().with("RECIPIENT", id)); return; } // [/RLVa:KB] // FIRE-8804: Prevent opening inventory from using share in radar context menu if (gRlvHandler.hasBehaviour(RLV_BHVR_SHOWINV)) { return; } // LLSD key; LLFloaterSidePanelContainer::showPanel("inventory", key); // [FS Communication UI] //LLFloaterReg::showInstance("im_container"); LLFloaterReg::showInstance("fs_im_container"); // [FS Communication UI] LLUUID session_id = gIMMgr->computeSessionID(IM_NOTHING_SPECIAL,id); if (!gIMMgr->hasSession(session_id)) { startIM(id); } if (gIMMgr->hasSession(session_id)) { // we should always get here, but check to verify anyways LLIMModel::getInstance()->addMessage(session_id, SYSTEM_FROM, LLUUID::null, LLTrans::getString("share_alert"), false); // [FS Communication UI] //LLFloaterIMSessionTab* session_floater = LLFloaterIMSessionTab::findConversation(session_id); //if (session_floater && session_floater->isMinimized()) //{ // session_floater->setMinimized(false); //} //LLFloaterIMContainer *im_container = LLFloaterReg::getTypedInstance("im_container"); //im_container->selectConversationPair(session_id, true); // [FS Communication UI] } } // Avatar tracking feature //static void LLAvatarActions::track(const LLUUID& id) { FSRadar* radar = FSRadar::getInstance(); if (radar) { radar->startTracking(id); } } // // Teleport to feature //static void LLAvatarActions::teleportTo(const LLUUID& id) { FSRadar* radar = FSRadar::getInstance(); if (radar) { radar->teleportToAvatar(id); } } namespace action_give_inventory { /** * Returns a pointer to 'Add More' inventory panel of Edit Outfit SP. */ static LLInventoryPanel* get_outfit_editor_inventory_panel() { LLPanelOutfitEdit* panel_outfit_edit = dynamic_cast(LLFloaterSidePanelContainer::getPanel("appearance", "panel_outfit_edit")); if (NULL == panel_outfit_edit) return NULL; LLInventoryPanel* inventory_panel = panel_outfit_edit->findChild("folder_view"); return inventory_panel; } /** * @return active inventory panel, or NULL if there's no such panel */ static LLInventoryPanel* get_active_inventory_panel() { LLInventoryPanel* active_panel = LLInventoryPanel::getActiveInventoryPanel(false); LLFloater* floater_appearance = LLFloaterReg::findInstance("appearance"); if (!active_panel || (floater_appearance && floater_appearance->hasFocus())) { active_panel = get_outfit_editor_inventory_panel(); } return active_panel; } /** * Checks My Inventory visibility. */ static bool is_give_inventory_acceptable_ids(const std::set inventory_selected_uuids) { if (inventory_selected_uuids.empty()) return false; // nothing selected bool acceptable = false; std::set::const_iterator it = inventory_selected_uuids.begin(); const std::set::const_iterator it_end = inventory_selected_uuids.end(); for (; it != it_end; ++it) { LLViewerInventoryCategory* inv_cat = gInventory.getCategory(*it); // any category can be offered. if (inv_cat) { acceptable = true; continue; } LLViewerInventoryItem* inv_item = gInventory.getItem(*it); // check if inventory item can be given if (LLGiveInventory::isInventoryGiveAcceptable(inv_item)) { acceptable = true; continue; } // there are neither item nor category in inventory acceptable = false; break; } return acceptable; } static bool is_give_inventory_acceptable(LLInventoryPanel* panel = NULL) { // check selection in the panel std::set inventory_selected_uuids = LLAvatarActions::getInventorySelectedUUIDs(panel); if (inventory_selected_uuids.empty()) { if(panel && panel->getRootFolder() && panel->getRootFolder()->isSingleFolderMode()) { inventory_selected_uuids.insert(panel->getRootFolderID()); } else { return false; // nothing selected } } return is_give_inventory_acceptable_ids(inventory_selected_uuids); } static void build_items_string(const std::set& inventory_selected_uuids , std::string& items_string) { llassert(inventory_selected_uuids.size() > 0); const std::string& separator = LLTrans::getString("words_separator"); for (std::set::const_iterator it = inventory_selected_uuids.begin(); ; ) { LLViewerInventoryCategory* inv_cat = gInventory.getCategory(*it); if (NULL != inv_cat) { items_string = inv_cat->getName(); break; } LLViewerInventoryItem* inv_item = gInventory.getItem(*it); if (NULL != inv_item) { items_string.append(inv_item->getName()); } if(++it == inventory_selected_uuids.end()) { break; } items_string.append(separator); } } struct LLShareInfo : public LLSingleton { LLSINGLETON_EMPTY_CTOR(LLShareInfo); public: std::vector mAvatarNames; uuid_vec_t mAvatarUuids; }; static void give_inventory_cb(const LLSD& notification, const LLSD& response, std::set inventory_selected_uuids) { S32 option = LLNotificationsUtil::getSelectedOption(notification, response); // if Cancel pressed if (option == 1) { return; } if (inventory_selected_uuids.empty()) { return; } auto count = LLShareInfo::instance().mAvatarNames.size(); bool shared = count && !inventory_selected_uuids.empty(); // iterate through avatars for(size_t i = 0; i < count; ++i) { const LLUUID& avatar_uuid = LLShareInfo::instance().mAvatarUuids[i]; // We souldn't open IM session, just calculate session ID for logging purpose. See EXT-6710 const LLUUID session_id = gIMMgr->computeSessionID(IM_NOTHING_SPECIAL, avatar_uuid); std::set::const_iterator it = inventory_selected_uuids.begin(); const std::set::const_iterator it_end = inventory_selected_uuids.end(); const std::string& separator = LLTrans::getString("words_separator"); std::string noncopy_item_names; LLSD noncopy_items = LLSD::emptyArray(); // iterate through selected inventory objects for (; it != it_end; ++it) { LLViewerInventoryCategory* inv_cat = gInventory.getCategory(*it); if (inv_cat) { if (!LLGiveInventory::doGiveInventoryCategory(avatar_uuid, inv_cat, session_id, "ItemsShared")) { shared = false; } break; } LLViewerInventoryItem* inv_item = gInventory.getItem(*it); if (!inv_item->getPermissions().allowCopyBy(gAgentID)) { if (!noncopy_item_names.empty()) { noncopy_item_names.append(separator); } noncopy_item_names.append(inv_item->getName()); noncopy_items.append(*it); } else { if (!LLGiveInventory::doGiveInventoryItem(avatar_uuid, inv_item, session_id)) { shared = false; } } } if (noncopy_items.beginArray() != noncopy_items.endArray()) { LLSD substitutions; substitutions["ITEMS"] = noncopy_item_names; LLSD payload; payload["agent_id"] = avatar_uuid; payload["items"] = noncopy_items; payload["success_notification"] = "ItemsShared"; LLNotificationsUtil::add("CannotCopyWarning", substitutions, payload, &LLGiveInventory::handleCopyProtectedItem); shared = false; break; } } if (shared) { LLFloaterReg::hideInstance("avatar_picker"); LLNotificationsUtil::add("ItemsShared"); } } /** * Performs "give inventory" operations for provided avatars. * * Sends one requests to give all selected inventory items for each passed avatar. * Avatars are represent by two vectors: names and UUIDs which must be sychronized with each other. * * @param avatar_names - avatar names request to be sent. * @param avatar_uuids - avatar names request to be sent. */ //static void give_inventory_ids(const uuid_vec_t& avatar_uuids, const std::vector avatar_names, const uuid_set_t inventory_selected_uuids) // [RLVa:KB] - @share static void give_inventory_ids(uuid_vec_t avatar_uuids, std::vector avatar_names, const uuid_set_t inventory_selected_uuids) // [/RLVa:KB] { llassert(avatar_names.size() == avatar_uuids.size()); // [RLVa:KB] - @share if ( (RlvActions::isRlvEnabled()) && (RlvActions::hasBehaviour(RLV_BHVR_SHARE)) ) { for (int idxAvatar = static_cast(avatar_uuids.size()) - 1; idxAvatar >= 0; idxAvatar--) { if (!RlvActions::canGiveInventory(avatar_uuids[idxAvatar])) { RlvUtil::notifyBlocked(RlvStringKeys::Blocked::Share, LLSD().with("RECIPIENT", LLSLURL("agent", avatar_uuids[idxAvatar], "completename").getSLURLString())); avatar_uuids.erase(avatar_uuids.begin() + idxAvatar); avatar_names.erase(avatar_names.begin() + idxAvatar); } } } if (avatar_uuids.empty()) { return; } // [/RLVa:KB] if (inventory_selected_uuids.empty()) { return; } std::string residents; LLAvatarActions::buildResidentsString(avatar_names, residents, true); std::string items; build_items_string(inventory_selected_uuids, items); int folders_count = 0; std::set::const_iterator it = inventory_selected_uuids.begin(); //traverse through selected inventory items and count folders among them for ( ; it != inventory_selected_uuids.end() && folders_count <=1 ; ++it) { LLViewerInventoryCategory* inv_cat = gInventory.getCategory(*it); if (NULL != inv_cat) { folders_count++; } } // EXP-1599 // In case of sharing multiple folders, make the confirmation // dialog contain a warning that only one folder can be shared at a time. std::string notification = (folders_count > 1) ? "ShareFolderConfirmation" : "ShareItemsConfirmation"; LLSD substitutions; substitutions["RESIDENTS"] = residents; substitutions["ITEMS"] = items; LLShareInfo::instance().mAvatarNames = avatar_names; LLShareInfo::instance().mAvatarUuids = avatar_uuids; LLNotificationsUtil::add(notification, substitutions, LLSD(), boost::bind(&give_inventory_cb, _1, _2, inventory_selected_uuids)); } static void give_inventory(const uuid_vec_t& avatar_uuids, const std::vector avatar_names, LLInventoryPanel* panel = NULL) { llassert(avatar_names.size() == avatar_uuids.size()); std::set inventory_selected_uuids = LLAvatarActions::getInventorySelectedUUIDs(panel);; if (inventory_selected_uuids.empty()) { if(panel && panel->getRootFolder() && panel->getRootFolder()->isSingleFolderMode()) { inventory_selected_uuids.insert(panel->getRootFolderID()); } else { return; } } give_inventory_ids(avatar_uuids, avatar_names, inventory_selected_uuids); } } // static void LLAvatarActions::buildResidentsString(std::vector avatar_names, std::string& residents_string, bool complete_name) { llassert(avatar_names.size() > 0); std::sort(avatar_names.begin(), avatar_names.end()); const std::string& separator = LLTrans::getString("words_separator"); for (std::vector::const_iterator it = avatar_names.begin(); ; ) { // FIRE-7923: Always show complete name when adding people to something! //if(complete_name) //{ // residents_string.append((*it).getCompleteName()); //} //else //{ // residents_string.append((*it).getDisplayName()); //} residents_string.append((*it).getCompleteName()); // if (++it == avatar_names.end()) { break; } residents_string.append(separator); } } // static void LLAvatarActions::buildResidentsString(const uuid_vec_t& avatar_uuids, std::string& residents_string) { std::vector avatar_names; uuid_vec_t::const_iterator it = avatar_uuids.begin(); for (; it != avatar_uuids.end(); ++it) { LLAvatarName av_name; if (LLAvatarNameCache::get(*it, &av_name)) { avatar_names.push_back(av_name); } } // We should check whether the vector is not empty to pass the assertion // that avatar_names.size() > 0 in LLAvatarActions::buildResidentsString. if (!avatar_names.empty()) { LLAvatarActions::buildResidentsString(avatar_names, residents_string); } } //static std::set LLAvatarActions::getInventorySelectedUUIDs(LLInventoryPanel* active_panel) { std::set inventory_selected; if (!active_panel) { active_panel = action_give_inventory::get_active_inventory_panel(); } if (active_panel) { inventory_selected= active_panel->getRootFolder()->getSelectionList(); } if (inventory_selected.empty()) { LLSidepanelInventory *sidepanel_inventory = LLFloaterSidePanelContainer::getPanel("inventory"); if (sidepanel_inventory) { inventory_selected= sidepanel_inventory->getInboxSelectionList(); } } std::set inventory_selected_uuids; for (std::set::iterator it = inventory_selected.begin(), end_it = inventory_selected.end(); it != end_it; ++it) { inventory_selected_uuids.insert(static_cast((*it)->getViewModelItem())->getUUID()); } return inventory_selected_uuids; } //static void LLAvatarActions::shareWithAvatars(LLView * panel) { // [RLVa:KB] - @share if ( (RlvActions::isRlvEnabled()) && (!RlvActions::canGiveInventory()) ) { RlvUtil::notifyBlocked(RlvStringKeys::Blocked::ShareGeneric); return; } // [/RLVa:KB] using namespace action_give_inventory; LLFloater* root_floater = gFloaterView->getParentFloater(panel); LLInventoryPanel* inv_panel = dynamic_cast(panel); LLFloaterAvatarPicker* picker = // FIRE-32377: Don't include own avatar when sharing items //LLFloaterAvatarPicker::show(boost::bind(give_inventory, _1, _2, inv_panel), true, false, false, root_floater->getName()); LLFloaterAvatarPicker::show(boost::bind(give_inventory, _1, _2, inv_panel), true, false, true, root_floater->getName()); if (!picker) { return; } picker->setOkBtnEnableCb(boost::bind(is_give_inventory_acceptable, inv_panel)); picker->openFriendsTab(); if (root_floater) { root_floater->addDependentFloater(picker); } LLNotificationsUtil::add("ShareNotification"); } //static void LLAvatarActions::shareWithAvatars(const uuid_set_t inventory_selected_uuids, LLFloater* root_floater) { using namespace action_give_inventory; LLFloaterAvatarPicker* picker = LLFloaterAvatarPicker::show(boost::bind(give_inventory_ids, _1, _2, inventory_selected_uuids), true, false, false, root_floater->getName()); if (!picker) { return; } picker->setOkBtnEnableCb(boost::bind(is_give_inventory_acceptable_ids, inventory_selected_uuids)); picker->openFriendsTab(); if (root_floater) { root_floater->addDependentFloater(picker); } LLNotificationsUtil::add("ShareNotification"); } // static bool LLAvatarActions::canShareSelectedItems(LLInventoryPanel* inv_panel /* = NULL*/) { using namespace action_give_inventory; if (!inv_panel) { LLInventoryPanel* active_panel = get_active_inventory_panel(); if (!active_panel) return false; inv_panel = active_panel; } // check selection in the panel LLFolderView* root_folder = inv_panel->getRootFolder(); if (!root_folder) { return false; } const std::set inventory_selected = root_folder->getSelectionList(); if (inventory_selected.empty()) return false; // nothing selected const LLUUID trash_id = gInventory.findCategoryUUIDForType(LLFolderType::FT_TRASH); bool can_share = true; std::set::const_iterator it = inventory_selected.begin(); const std::set::const_iterator it_end = inventory_selected.end(); for (; it != it_end; ++it) { LLUUID cat_id = static_cast((*it)->getViewModelItem())->getUUID(); LLViewerInventoryCategory* inv_cat = gInventory.getCategory(cat_id); // any category can be offered if it's not in trash. if (inv_cat) { if ((cat_id == trash_id) || gInventory.isObjectDescendentOf(cat_id, trash_id)) { can_share = false; break; } continue; } // check if inventory item can be given LLFolderViewItem* item = *it; if (!item) return false; LLInvFVBridge* bridge = dynamic_cast(item->getViewModelItem()); if (bridge && bridge->canShare()) { continue; } // there are neither item nor category in inventory can_share = false; break; } return can_share; } // static bool LLAvatarActions::toggleBlock(const LLUUID& id) { LLAvatarName av_name; LLAvatarNameCache::get(id, &av_name); LLMute mute(id, av_name.getUserName(), LLMute::AGENT); if (LLMuteList::getInstance()->isMuted(mute.mID, mute.mName)) { LLMuteList::getInstance()->remove(mute); return false; } else { LLMuteList::getInstance()->add(mute); return true; } } // static void LLAvatarActions::toggleMute(const LLUUID& id, U32 flags) { LLAvatarName av_name; LLAvatarNameCache::get(id, &av_name); LLMuteList* mute_list = LLMuteList::getInstance(); bool is_muted = mute_list->isMuted(id, flags); LLMute mute(id, av_name.getUserName(), LLMute::AGENT); if (!is_muted) { mute_list->add(mute, flags); } else { mute_list->remove(mute, flags); } } // static void LLAvatarActions::toggleMuteVoice(const LLUUID& id) { toggleMute(id, LLMute::flagVoiceChat); } // static bool LLAvatarActions::canOfferTeleport(const LLUUID& id) { // RLV support // Only allow offering teleports if everyone is a @tplure exception or able to map this avie under @showloc=n if (gRlvHandler.hasBehaviour(RLV_BHVR_SHOWLOC)) { const LLRelationship* pBuddyInfo = LLAvatarTracker::instance().getBuddyInfo(id); if (gRlvHandler.isException(RLV_BHVR_TPLURE, id, ERlvExceptionCheck::Permissive) || (pBuddyInfo && pBuddyInfo->isOnline() && pBuddyInfo->isRightGrantedTo(LLRelationship::GRANT_MAP_LOCATION))) { return true; } return false; } // // First use LLAvatarTracker::isBuddy() // If LLAvatarTracker::instance().isBuddyOnline function only is used // then for avatars that are online and not a friend it will return false. // But we should give an ability to offer a teleport for such avatars. if(LLAvatarTracker::instance().isBuddy(id)) { return LLAvatarTracker::instance().isBuddyOnline(id); } return true; } // static bool LLAvatarActions::canOfferTeleport(const uuid_vec_t& ids) { // We can't send more than 250 lures in a single message, so disable this // button when there are too many id's selected. // Fix edge case with multi-selection containing offlines //if(ids.size() > 250) return false; // //bool result = true; //for (uuid_vec_t::const_iterator it = ids.begin(); it != ids.end(); ++it) //{ // if(!canOfferTeleport(*it)) // { // result = false; // break; // } //} //return result; if (ids.size() == 1) { return canOfferTeleport(ids.front()); } S32 valid_count = 0; for (uuid_vec_t::const_iterator it = ids.begin(); it != ids.end(); ++it) { if (canOfferTeleport(*it)) { valid_count++; } } return (valid_count > 0 && valid_count <= 250); // } // Extra request teleport // static bool LLAvatarActions::canRequestTeleport(const LLUUID& id) { if(LLAvatarTracker::instance().isBuddy(id)) { return LLAvatarTracker::instance().isBuddyOnline(id); } return true; } void LLAvatarActions::inviteToGroup(const LLUUID& id) { LLFloaterGroupPicker* widget = LLFloaterReg::showTypedInstance("group_picker", LLSD(id)); if (widget) { widget->center(); widget->setPowersMask(GP_MEMBER_INVITE); widget->removeNoneOption(); widget->setSelectGroupCallback(boost::bind(callback_invite_to_group, _1, id)); } } // static void LLAvatarActions::viewChatHistory(const LLUUID& id) { const std::vector& conversations = LLConversationLog::instance().getConversations(); std::vector::const_iterator iter = conversations.begin(); for (; iter != conversations.end(); ++iter) { if (iter->getParticipantID() == id) { // External chat history option //LLFloaterReg::showInstance("preview_conversation", iter->getSessionID(), true); if (gSavedSettings.getBOOL("FSUseBuiltInHistory")) { LLFloaterReg::showInstance("preview_conversation", iter->getSessionID(), true); } else { gViewerWindow->getWindow()->openFile(LLLogChat::makeLogFileName(iter->getHistoryFileName())); } // return; } } if (LLLogChat::isTranscriptExist(id)) { LLAvatarName avatar_name; LLSD extended_id(id); LLAvatarNameCache::get(id, &avatar_name); extended_id[LL_FCP_COMPLETE_NAME] = avatar_name.getCompleteName(); // [Legacy IM logfile names] //extended_id[LL_FCP_ACCOUNT_NAME] = avatar_name.getAccountName(); //LLFloaterReg::showInstance("preview_conversation", extended_id, true); if (gSavedSettings.getBOOL("UseLegacyIMLogNames")) { std::string avatar_user_name = avatar_name.getUserName(); extended_id[LL_FCP_ACCOUNT_NAME] = avatar_user_name.substr(0, avatar_user_name.find(" Resident"));; } else { extended_id[LL_FCP_ACCOUNT_NAME] = avatar_name.getAccountName(); } if (gSavedSettings.getBOOL("FSUseBuiltInHistory")) { LLFloaterReg::showInstance("preview_conversation", extended_id, true); } else { gViewerWindow->getWindow()->openFile(LLLogChat::makeLogFileName(extended_id[LL_FCP_ACCOUNT_NAME].asString())); } // [Legacy IM logfile names] } } // Open chat history externally // static void LLAvatarActions::viewChatHistoryExternally(const LLUUID& id) { if (LLLogChat::isTranscriptExist(id)) { LLAvatarName av_name; if (LLAvatarNameCache::get(id, &av_name)) { std::string history_filename; if (gSavedSettings.getBOOL("UseLegacyIMLogNames")) { std::string user_name = av_name.getUserName(); history_filename = user_name.substr(0, user_name.find(" Resident"));; } else { history_filename = LLCacheName::buildUsername(av_name.getUserName()); } gViewerWindow->getWindow()->openFile(LLLogChat::makeLogFileName(history_filename)); } } } // // [FS:CR] Add to contact set //static void LLAvatarActions::addToContactSet(const LLUUID& agent_id) { LLFloaterReg::showInstance("fs_add_contact", agent_id, true); } void LLAvatarActions::addToContactSet(const uuid_vec_t& agent_ids) { if (agent_ids.size() == 1) { LLAvatarActions::addToContactSet(agent_ids.front()); } else { LLSD data; for (uuid_vec_t::const_iterator it = agent_ids.begin(); it != agent_ids.end(); ++it) { data.append(*it); } LLFloaterReg::showInstance("fs_add_contact", data, true); } } // [/FS:CR] Add to contact set //== private methods ======================================================================================== // static bool LLAvatarActions::handleRemove(const LLSD& notification, const LLSD& response) { S32 option = LLNotificationsUtil::getSelectedOption(notification, response); const LLSD& ids = notification["payload"]["ids"]; for (LLSD::array_const_iterator itr = ids.beginArray(); itr != ids.endArray(); ++itr) { LLUUID id = itr->asUUID(); const LLRelationship* ip = LLAvatarTracker::instance().getBuddyInfo(id); if (ip) { switch (option) { case 0: // YES if( ip->isRightGrantedTo(LLRelationship::GRANT_MODIFY_OBJECTS)) { LLAvatarTracker::instance().empower(id, false); LLAvatarTracker::instance().notifyObservers(); } LLAvatarTracker::instance().terminateBuddy(id); LLAvatarTracker::instance().notifyObservers(); break; case 1: // NO default: LL_INFOS() << "No removal performed." << LL_ENDL; break; } } } return false; } // static bool LLAvatarActions::handlePay(const LLSD& notification, const LLSD& response, LLUUID avatar_id) { S32 option = LLNotificationsUtil::getSelectedOption(notification, response); if (option == 0) { gAgent.setDoNotDisturb(false); } LLFloaterPayUtil::payDirectly(&give_money, avatar_id, /*is_group=*/false); return false; } // static void LLAvatarActions::callback_invite_to_group(LLUUID group_id, LLUUID id) { uuid_vec_t agent_ids; agent_ids.push_back(id); LLFloaterGroupInvite::showForGroup(group_id, &agent_ids); } // static bool LLAvatarActions::callbackAddFriendWithMessage(const LLSD& notification, const LLSD& response) { S32 option = LLNotificationsUtil::getSelectedOption(notification, response); if (option == 0) { requestFriendship(notification["payload"]["id"].asUUID(), notification["payload"]["name"].asString(), response["message"].asString()); } return false; } // static bool LLAvatarActions::handleKick(const LLSD& notification, const LLSD& response) { S32 option = LLNotification::getSelectedOption(notification, response); if (option == 0) { LLUUID avatar_id = notification["payload"]["avatar_id"].asUUID(); LLMessageSystem* msg = gMessageSystem; msg->newMessageFast(_PREHASH_GodKickUser); msg->nextBlockFast(_PREHASH_UserInfo); msg->addUUIDFast(_PREHASH_GodID, gAgent.getID() ); msg->addUUIDFast(_PREHASH_GodSessionID, gAgent.getSessionID()); msg->addUUIDFast(_PREHASH_AgentID, avatar_id ); msg->addU32("KickFlags", KICK_FLAGS_DEFAULT ); msg->addStringFast(_PREHASH_Reason, response["message"].asString() ); gAgent.sendReliableMessage(); } return false; } bool LLAvatarActions::handleFreezeAvatar(const LLSD& notification, const LLSD& response) { S32 option = LLNotification::getSelectedOption(notification, response); if (0 == option || 1 == option) { U32 flags = 0x0; if (1 == option) { // unfreeze flags |= 0x1; } LLUUID avatar_id = notification["payload"]["avatar_id"].asUUID(); LLMessageSystem* msg = gMessageSystem; msg->newMessage("FreezeUser"); msg->nextBlock("AgentData"); msg->addUUID("AgentID", gAgent.getID()); msg->addUUID("SessionID", gAgent.getSessionID()); msg->nextBlock("Data"); msg->addUUID("TargetID", avatar_id ); msg->addU32("Flags", flags ); gAgent.sendReliableMessage(); } return false; } bool LLAvatarActions::handleEjectAvatar(const LLSD& notification, const LLSD& response) { S32 option = LLNotificationsUtil::getSelectedOption(notification, response); if (2 == option) { return false; } LLUUID avatar_id = notification["payload"]["avatar_id"].asUUID(); bool ban_enabled = notification["payload"]["ban_enabled"].asBoolean(); if (0 == option) { LLMessageSystem* msg = gMessageSystem; U32 flags = 0x0; msg->newMessage("EjectUser"); msg->nextBlock("AgentData"); msg->addUUID("AgentID", gAgent.getID() ); msg->addUUID("SessionID", gAgent.getSessionID() ); msg->nextBlock("Data"); msg->addUUID("TargetID", avatar_id ); msg->addU32("Flags", flags ); gAgent.sendReliableMessage(); } else if (ban_enabled) { LLMessageSystem* msg = gMessageSystem; U32 flags = 0x1; msg->newMessage("EjectUser"); msg->nextBlock("AgentData"); msg->addUUID("AgentID", gAgent.getID() ); msg->addUUID("SessionID", gAgent.getSessionID() ); msg->nextBlock("Data"); msg->addUUID("TargetID", avatar_id ); msg->addU32("Flags", flags ); gAgent.sendReliableMessage(); } return false; } bool LLAvatarActions::handleFreeze(const LLSD& notification, const LLSD& response) { S32 option = LLNotification::getSelectedOption(notification, response); if (option == 0) { LLUUID avatar_id = notification["payload"]["avatar_id"].asUUID(); LLMessageSystem* msg = gMessageSystem; msg->newMessageFast(_PREHASH_GodKickUser); msg->nextBlockFast(_PREHASH_UserInfo); msg->addUUIDFast(_PREHASH_GodID, gAgent.getID() ); msg->addUUIDFast(_PREHASH_GodSessionID, gAgent.getSessionID()); msg->addUUIDFast(_PREHASH_AgentID, avatar_id ); msg->addU32("KickFlags", KICK_FLAGS_FREEZE ); msg->addStringFast(_PREHASH_Reason, response["message"].asString() ); gAgent.sendReliableMessage(); } return false; } bool LLAvatarActions::handleUnfreeze(const LLSD& notification, const LLSD& response) { S32 option = LLNotification::getSelectedOption(notification, response); std::string text = response["message"].asString(); if (option == 0) { LLUUID avatar_id = notification["payload"]["avatar_id"].asUUID(); LLMessageSystem* msg = gMessageSystem; msg->newMessageFast(_PREHASH_GodKickUser); msg->nextBlockFast(_PREHASH_UserInfo); msg->addUUIDFast(_PREHASH_GodID, gAgent.getID() ); msg->addUUIDFast(_PREHASH_GodSessionID, gAgent.getSessionID()); msg->addUUIDFast(_PREHASH_AgentID, avatar_id ); msg->addU32("KickFlags", KICK_FLAGS_UNFREEZE ); msg->addStringFast(_PREHASH_Reason, text ); gAgent.sendReliableMessage(); } return false; } // static void LLAvatarActions::requestFriendship(const LLUUID& target_id, const std::string& target_name, const std::string& message) { const LLUUID calling_card_folder_id = gInventory.findCategoryUUIDForType(LLFolderType::FT_CALLINGCARD); LLUIUsage::instance().logCommand("Agent.SendFriendRequest"); send_improved_im(target_id, target_name, message, IM_ONLINE, IM_FRIENDSHIP_OFFERED, calling_card_folder_id); LLSD args; args["TO_NAME"] = target_name; LLSD payload; payload["from_id"] = target_id; LLNotificationsUtil::add("FriendshipOffered", args, payload); } //static bool LLAvatarActions::isFriend(const LLUUID& id) { return ( NULL != LLAvatarTracker::instance().getBuddyInfo(id) ); } // static bool LLAvatarActions::isBlocked(const LLUUID& id) { LLAvatarName av_name; LLAvatarNameCache::get(id, &av_name); return LLMuteList::getInstance()->isMuted(id, av_name.getUserName()); } // static bool LLAvatarActions::isVoiceMuted(const LLUUID& id) { return LLMuteList::getInstance()->isMuted(id, LLMute::flagVoiceChat); } // static bool LLAvatarActions::canBlock(const LLUUID& id) { LLAvatarName av_name; LLAvatarNameCache::get(id, &av_name); std::string full_name = av_name.getUserName(); bool is_linden = (full_name.find("Linden") != std::string::npos); bool is_self = id == gAgentID; return !is_self && !is_linden; } // [SL:KB] - Patch: UI-SidepanelPeople | Checked: 2010-12-03 (Catznip-2.4.0g) | Modified: Catznip-2.4.0g void LLAvatarActions::report(const LLUUID& idAgent) { LLAvatarName avName; LLAvatarNameCache::get(idAgent, &avName); LLFloaterReporter::showFromAvatar(idAgent, avName.getCompleteName()); } bool LLAvatarActions::canZoomIn(const LLUUID& idAgent) { // Firestorm radar support //return gObjectList.findObject(idAgent); LLViewerObject* object = gObjectList.findObject(idAgent); if (object) { return true; } else { // Special case for SL since interest list changes auto entry = FSRadar::getInstance()->getEntry(idAgent); #ifdef OPENSIM if (LLGridManager::getInstance()->isInOpenSim()) { return (entry && entry->getRange() <= gSavedSettings.getF32("RenderFarClip")); } else #endif { return (entry != nullptr); } } // } void LLAvatarActions::zoomIn(const LLUUID& idAgent) { // Firestorm radar support //handle_zoom_to_object(idAgent); if (auto entry = FSRadar::getInstance()->getEntry(idAgent); entry) { handle_zoom_to_object(idAgent, entry->getGlobalPos()); } else { handle_zoom_to_object(idAgent); } // } void LLAvatarActions::getScriptInfo(const LLUUID& idAgent) { LL_INFOS() << "Reporting Script Info for avatar: " << idAgent.asString() << LL_ENDL; FSLSLBridge::instance().viewerToLSL("getScriptInfo|" + idAgent.asString() + "|" + (gSavedSettings.getBOOL("FSScriptInfoExtended") ? "1" : "0")); } // // Parcel actions // // Defined in llworld.cpp LLVector3d unpackLocalToGlobalPosition(U32 compact_local, const LLVector3d& region_origin, F32 width_scale_factor); // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g bool getRegionAndPosGlobalFromAgentID(const LLUUID& idAgent, const LLViewerRegion** ppRegion, LLVector3d* pPosGlobal) { // Try looking up the agent in gObjectList const LLViewerObject* pAvatarObj = gObjectList.findObject(idAgent); if (pAvatarObj) { if (ppRegion) *ppRegion = pAvatarObj->getRegion(); if (pPosGlobal) *pPosGlobal = pAvatarObj->getPositionGlobal(); return (pAvatarObj->isAvatar()) && (NULL != pAvatarObj->getRegion()); } // Walk over each region we're connected to and try finding the agent on one of them LLWorld::region_list_t::const_iterator itRegion = LLWorld::getInstance()->getRegionList().begin(); LLWorld::region_list_t::const_iterator endRegion = LLWorld::getInstance()->getRegionList().end(); for (; itRegion != endRegion; ++itRegion) { const LLViewerRegion* pRegion = *itRegion; for (size_t idxRegionAgent = 0, cntRegionAgent = pRegion->mMapAvatars.size(); idxRegionAgent < cntRegionAgent; idxRegionAgent++) { if (pRegion->mMapAvatarIDs.at(idxRegionAgent) == idAgent) { if (ppRegion) *ppRegion = pRegion; if (pPosGlobal) *pPosGlobal = unpackLocalToGlobalPosition(pRegion->mMapAvatars.at(idxRegionAgent), pRegion->getOriginGlobal(), pRegion->getWidthScaleFactor()); return (NULL != pRegion); } } } // Couldn't find the agent anywhere return false; } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g inline bool getRegionFromAgentID(const LLUUID& idAgent, const LLViewerRegion** ppRegion) { return getRegionAndPosGlobalFromAgentID(idAgent, ppRegion, NULL); } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g inline bool getPosGlobalFromAgentID(const LLUUID& idAgent, LLVector3d& posGlobal) { return getRegionAndPosGlobalFromAgentID(idAgent, NULL, &posGlobal); } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g bool LLAvatarActions::canLandFreezeOrEject(const LLUUID& idAgent) { uuid_vec_t idAgents; idAgents.push_back(idAgent); return canLandFreezeOrEjectMultiple(idAgents); } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g bool LLAvatarActions::canLandFreezeOrEjectMultiple(uuid_vec_t& idAgents, bool fFilter /*=false*/) { if (gAgent.isGodlikeWithoutAdminMenuFakery()) return true; // Gods can always freeze uuid_vec_t::iterator itAgent = idAgents.begin(); bool fCanFreeze = false; while ( (itAgent != idAgents.end()) && ((fFilter) || (!fCanFreeze)) ) { const LLViewerRegion* pRegion = NULL; LLVector3d posGlobal; if (getRegionAndPosGlobalFromAgentID(*itAgent, &pRegion, &posGlobal)) { // NOTE: we actually don't always need the parcel, but attempting to get it now will help with setting fBanEnabled when ejecting const LLParcel* pParcel = LLViewerParcelMgr::getInstance()->selectParcelAt(posGlobal)->getParcel(); const LLVector3 posRegion = pRegion->getPosRegionFromGlobal(posGlobal); if ( (pRegion->getOwner() == gAgent.getID()) || (pRegion->isEstateManager()) || (pRegion->isOwnedSelf(posRegion)) || ((pRegion->isOwnedGroup(posRegion)) && (pParcel) && (gAgent.hasPowerInGroup(pParcel->getOwnerID(), GP_LAND_ADMIN))) ) { fCanFreeze = true; ++itAgent; continue; } } if (fFilter) itAgent = idAgents.erase(itAgent); else ++itAgent; } return fCanFreeze; } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g void LLAvatarActions::landEject(const LLUUID& idAgent) { LL_INFOS() << "landeject " << idAgent << LL_ENDL; uuid_vec_t idAgents; idAgents.push_back(idAgent); landEjectMultiple(idAgents); } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g void LLAvatarActions::landEjectMultiple(const uuid_vec_t& idAgents) { uuid_vec_t idEjectAgents(idAgents); if (!canLandFreezeOrEjectMultiple(idEjectAgents, true)) { LL_WARNS() << "Not allowed to eject" << LL_ENDL; return; } LLSD args, payload; std::string strMsgName, strResidents; bool fBanEnabled = false; for (uuid_vec_t::const_iterator itAgent = idEjectAgents.begin(); itAgent != idEjectAgents.end(); ++itAgent) { const LLUUID& idAgent = *itAgent; LLVector3d posGlobal; if ( (!fBanEnabled) && (getPosGlobalFromAgentID(idAgent, posGlobal)) ) { const LLParcel* pParcel = LLViewerParcelMgr::getInstance()->selectParcelAt(posGlobal)->getParcel(); fBanEnabled = (pParcel) && (LLViewerParcelMgr::getInstance()->isParcelOwnedByAgent(pParcel, GP_LAND_MANAGE_BANNED)); } if (idEjectAgents.begin() != itAgent) strResidents += "\n"; strResidents += LLSLURL("agent", idAgent, (!gRlvHandler.hasBehaviour(RLV_BHVR_SHOWNAMES)) ? "completename" : "rlvanonym").getSLURLString(); payload["ids"].append(*itAgent); } if (1 == payload["ids"].size()) { args["AVATAR_NAME"] = strResidents; strMsgName = (fBanEnabled) ? "EjectAvatarFullname" : "EjectAvatarFullnameNoBan"; } else { args["RESIDENTS"] = strResidents; strMsgName = (fBanEnabled) ? "EjectAvatarMultiple" : "EjectAvatarMultipleNoBan"; } payload["ban_enabled"] = fBanEnabled; LLNotificationsUtil::add(strMsgName, args, payload, &callbackLandEject); } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g bool LLAvatarActions::callbackLandEject(const LLSD& notification, const LLSD& response) { S32 idxOption = LLNotificationsUtil::getSelectedOption(notification, response); if (2 == idxOption) // Cancel button. return false; bool fBanEnabled = notification["payload"]["ban_enabled"].asBoolean(); if ( (0 == idxOption) || (fBanEnabled) ) // Eject button (or Eject + Ban) { const LLSD& idAgents = notification["payload"]["ids"]; for (LLSD::array_const_iterator itAgent = idAgents.beginArray(); itAgent != idAgents.endArray(); ++itAgent) { const LLUUID idAgent = itAgent->asUUID(); const LLViewerRegion* pAgentRegion = NULL; if (getRegionFromAgentID(idAgent, &pAgentRegion)) { // This is tricky. It is similar to say if it is not an 'Eject' button, and it is also not an 'Cancel' button, // and ban_enabled==true, it should be the 'Eject and Ban' button. U32 flags = ( (0 != idxOption) && (fBanEnabled) ) ? 0x1 : 0x0; gMessageSystem->newMessage("EjectUser"); gMessageSystem->nextBlock("AgentData"); gMessageSystem->addUUID("AgentID", gAgent.getID()); gMessageSystem->addUUID("SessionID", gAgent.getSessionID()); gMessageSystem->nextBlock("Data"); gMessageSystem->addUUID("TargetID", idAgent); gMessageSystem->addU32("Flags", flags); gMessageSystem->sendReliable(pAgentRegion->getHost()); } } } return false; } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g void LLAvatarActions::landFreeze(const LLUUID& idAgent) { LL_INFOS() << "landfreezing " << idAgent << LL_ENDL; uuid_vec_t idAgents; idAgents.push_back(idAgent); landFreezeMultiple(idAgents); } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g void LLAvatarActions::landFreezeMultiple(const uuid_vec_t& idAgents) { uuid_vec_t idEjectAgents(idAgents); if (!canLandFreezeOrEjectMultiple(idEjectAgents, true)) return; LLSD args, payload; std::string strMsgName, strResidents; for (uuid_vec_t::const_iterator itAgent = idEjectAgents.begin(); itAgent != idEjectAgents.end(); ++itAgent) { const LLUUID& idAgent = *itAgent; if (idEjectAgents.begin() != itAgent) strResidents += "\n"; strResidents += LLSLURL("agent", idAgent, (!gRlvHandler.hasBehaviour(RLV_BHVR_SHOWNAMES)) ? "completename" : "rlvanonym").getSLURLString(); payload["ids"].append(*itAgent); } if (1 == payload["ids"].size()) { args["AVATAR_NAME"] = strResidents; strMsgName = "FreezeAvatarFullname"; } else { args["RESIDENTS"] = strResidents; strMsgName = "FreezeAvatarMultiple"; } LLNotificationsUtil::add(strMsgName, args, payload, &callbackLandFreeze); } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g bool LLAvatarActions::callbackLandFreeze(const LLSD& notification, const LLSD& response) { S32 idxOption = LLNotificationsUtil::getSelectedOption(notification, response); if ( (0 == idxOption) || (1 == idxOption) ) { U32 flags = (0 == idxOption) ? 0x0 : 0x1; const LLSD& idAgents = notification["payload"]["ids"]; for (LLSD::array_const_iterator itAgent = idAgents.beginArray(); itAgent != idAgents.endArray(); ++itAgent) { const LLUUID idAgent = itAgent->asUUID(); const LLViewerRegion* pAgentRegion = NULL; if (getRegionFromAgentID(idAgent, &pAgentRegion)) { gMessageSystem->newMessage("FreezeUser"); gMessageSystem->nextBlock("AgentData"); gMessageSystem->addUUID("AgentID", gAgent.getID()); gMessageSystem->addUUID("SessionID", gAgent.getSessionID()); gMessageSystem->nextBlock("Data"); gMessageSystem->addUUID("TargetID", idAgent); gMessageSystem->addU32("Flags", flags); gMessageSystem->sendReliable(pAgentRegion->getHost()); } } } return false; } // // Estate actions // typedef std::vector strings_t; // Copy/paste from LLPanelRegionInfo::sendEstateOwnerMessage void sendEstateOwnerMessage(const LLViewerRegion* pRegion, const std::string& request, const LLUUID& invoice, const strings_t& strings) { if (pRegion) { LL_INFOS() << "Sending estate request '" << request << "'" << LL_ENDL; gMessageSystem->newMessage("EstateOwnerMessage"); gMessageSystem->nextBlockFast(_PREHASH_AgentData); gMessageSystem->addUUIDFast(_PREHASH_AgentID, gAgent.getID()); gMessageSystem->addUUIDFast(_PREHASH_SessionID, gAgent.getSessionID()); gMessageSystem->addUUIDFast(_PREHASH_TransactionID, LLUUID::null); //not used gMessageSystem->nextBlock("MethodData"); gMessageSystem->addString("Method", request); gMessageSystem->addUUID("Invoice", invoice); if(strings.empty()) { gMessageSystem->nextBlock("ParamList"); gMessageSystem->addString("Parameter", NULL); } else { strings_t::const_iterator it = strings.begin(); strings_t::const_iterator end = strings.end(); for(; it != end; ++it) { gMessageSystem->nextBlock("ParamList"); gMessageSystem->addString("Parameter", *it); } } gMessageSystem->sendReliable(pRegion->getHost()); } } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g bool LLAvatarActions::canEstateKickOrTeleportHome(const LLUUID& idAgent) { uuid_vec_t idAgents; idAgents.push_back(idAgent); return canEstateKickOrTeleportHomeMultiple(idAgents); } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g bool LLAvatarActions::canEstateKickOrTeleportHomeMultiple(uuid_vec_t& idAgents, bool fFilter /*=false*/) { if (gAgent.isGodlikeWithoutAdminMenuFakery()) return true; // Gods can always kick uuid_vec_t::iterator itAgent = idAgents.begin(); bool fCanKick = false; while ( (itAgent != idAgents.end()) && ((fFilter) || (!fCanKick)) ) { const LLViewerRegion* pRegion = NULL; if ( (getRegionFromAgentID(*itAgent, &pRegion)) && ((pRegion->getOwner() == gAgent.getID()) || (pRegion->isEstateManager())) ) { fCanKick = true; ++itAgent; // Estate owners/managers can kick continue; } if (fFilter) itAgent = idAgents.erase(itAgent); else ++itAgent; } return fCanKick; } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g void LLAvatarActions::estateKick(const LLUUID& idAgent) { LL_INFOS() << "estatekick " << idAgent << LL_ENDL; uuid_vec_t idAgents; idAgents.push_back(idAgent); estateKickMultiple(idAgents); } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g void LLAvatarActions::estateKickMultiple(const uuid_vec_t& idAgents) { uuid_vec_t idEjectAgents(idAgents); if (!canEstateKickOrTeleportHomeMultiple(idEjectAgents, true)) return; LLSD args, payload; std::string strMsgName, strResidents; for (uuid_vec_t::const_iterator itAgent = idEjectAgents.begin(); itAgent != idEjectAgents.end(); ++itAgent) { const LLUUID& idAgent = *itAgent; if (idEjectAgents.begin() != itAgent) strResidents += "\n"; strResidents += LLSLURL("agent", idAgent, (!gRlvHandler.hasBehaviour(RLV_BHVR_SHOWNAMES)) ? "completename" : "rlvanonym").getSLURLString(); payload["ids"].append(*itAgent); } if (1 == payload["ids"].size()) { args["EVIL_USER"] = strResidents; strMsgName = "EstateKickUser"; } else { args["RESIDENTS"] = strResidents; strMsgName = "EstateKickMultiple"; } LLNotificationsUtil::add(strMsgName, args, payload, &callbackEstateKick); } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g bool LLAvatarActions::callbackEstateKick(const LLSD& notification, const LLSD& response) { S32 idxOption = LLNotificationsUtil::getSelectedOption(notification, response); if (0 == idxOption) { const LLSD& idAgents = notification["payload"]["ids"]; for (LLSD::array_const_iterator itAgent = idAgents.beginArray(); itAgent != idAgents.endArray(); ++itAgent) { const LLViewerRegion* pRegion = NULL; if (getRegionFromAgentID(itAgent->asUUID(), &pRegion)) { strings_t strings; strings.push_back(itAgent->asString()); sendEstateOwnerMessage(pRegion, "kickestate", LLUUID::generateNewID(), strings); } } } return false; } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g void LLAvatarActions::estateTeleportHome(const LLUUID& idAgent) { LL_INFOS() << "estateTpHome " << idAgent << LL_ENDL; uuid_vec_t idAgents; idAgents.push_back(idAgent); estateTeleportHomeMultiple(idAgents); } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g void LLAvatarActions::estateTeleportHomeMultiple(const uuid_vec_t& idAgents) { uuid_vec_t idEjectAgents(idAgents); if (!canEstateKickOrTeleportHomeMultiple(idEjectAgents, true)) return; LLSD args, payload; std::string strMsgName, strResidents; for (uuid_vec_t::const_iterator itAgent = idEjectAgents.begin(); itAgent != idEjectAgents.end(); ++itAgent) { const LLUUID& idAgent = *itAgent; if (idEjectAgents.begin() != itAgent) strResidents += "\n"; strResidents += LLSLURL("agent", idAgent, (!gRlvHandler.hasBehaviour(RLV_BHVR_SHOWNAMES)) ? "completename" : "rlvanonym").getSLURLString(); payload["ids"].append(*itAgent); } if (1 == payload["ids"].size()) { args["AVATAR_NAME"] = strResidents; strMsgName = "EstateTeleportHomeUser"; } else { args["RESIDENTS"] = strResidents; strMsgName = "EstateTeleportHomeMultiple"; } LLNotificationsUtil::add(strMsgName, args, payload, &callbackEstateTeleportHome); } // static - Checked: 2010-12-03 (Catznip-2.4.0g) | Added: Catznip-2.4.0g bool LLAvatarActions::callbackEstateTeleportHome(const LLSD& notification, const LLSD& response) { S32 idxOption = LLNotificationsUtil::getSelectedOption(notification, response); if (0 == idxOption) { const LLSD& idAgents = notification["payload"]["ids"]; for (LLSD::array_const_iterator itAgent = idAgents.beginArray(); itAgent != idAgents.endArray(); ++itAgent) { const LLViewerRegion* pRegion = NULL; if (getRegionFromAgentID(itAgent->asUUID(), &pRegion)) { strings_t strings; strings.push_back(gAgent.getID().asString()); strings.push_back(itAgent->asString()); sendEstateOwnerMessage(pRegion, "teleporthomeuser", LLUUID::generateNewID(), strings); } } } return false; } // [/SL:KB] // Estate ban user void LLAvatarActions::estateBan(const LLUUID& idAgent) { uuid_vec_t idAgents; idAgents.push_back(idAgent); estateBanMultiple(idAgents); } void LLAvatarActions::estateBanMultiple(const uuid_vec_t& idAgents) { LLViewerRegion* region = gAgent.getRegion(); if (!region) { return; } uuid_vec_t idEjectAgents(idAgents); if (!canEstateKickOrTeleportHomeMultiple(idEjectAgents, true)) return; LLSD args, payload; std::string strMsgName, strResidents; for (uuid_vec_t::const_iterator itAgent = idEjectAgents.begin(); itAgent != idEjectAgents.end(); ++itAgent) { const LLUUID& idAgent = *itAgent; if (idEjectAgents.begin() != itAgent) strResidents += "\n"; strResidents += LLSLURL("agent", idAgent, (!gRlvHandler.hasBehaviour(RLV_BHVR_SHOWNAMES)) ? "completename" : "rlvanonym").getSLURLString(); payload["ids"].append(*itAgent); } std::string owner = LLSLURL("agent", region->getOwner(), "inspect").getSLURLString(); if (gAgent.isGodlike()) { LLStringUtil::format_map_t owner_args; owner_args["[OWNER]"] = owner; args["ALL_ESTATES"] = LLTrans::getString("RegionInfoAllEstatesOwnedBy", owner_args); } else if (region->getOwner() == gAgent.getID()) { args["ALL_ESTATES"] = LLTrans::getString("RegionInfoAllEstatesYouOwn"); } else if (region->isEstateManager()) { LLStringUtil::format_map_t owner_args; owner_args["[OWNER]"] = owner.c_str(); args["ALL_ESTATES"] = LLTrans::getString("RegionInfoAllEstatesYouManage", owner_args); } if (1 == payload["ids"].size()) { args["EVIL_USER"] = strResidents; strMsgName = "EstateBanUser"; } else { args["RESIDENTS"] = strResidents; strMsgName = "EstateBanUserMultiple"; } LLNotificationsUtil::add(strMsgName, args, payload, &callbackEstateBan); } bool LLAvatarActions::callbackEstateBan(const LLSD& notification, const LLSD& response) { LLViewerRegion* region = gAgent.getRegion(); S32 idxOption = LLNotificationsUtil::getSelectedOption(notification, response); if (0 == idxOption || 1 == idxOption) { const LLSD& idAgents = notification["payload"]["ids"]; for (LLSD::array_const_iterator itAgent = idAgents.beginArray(); itAgent != idAgents.endArray(); ++itAgent) { if (region->getOwner() == itAgent->asUUID()) { // Can't ban the owner! continue; } U32 flags = ESTATE_ACCESS_BANNED_AGENT_ADD | ESTATE_ACCESS_ALLOWED_AGENT_REMOVE; if (itAgent + 1 != idAgents.endArray()) { flags |= ESTATE_ACCESS_NO_REPLY; } if (idxOption == 1) { // All estates, either than I own or manage for this owner. // This will be verified on simulator. JC if (!region) { break; } if (region->getOwner() == gAgent.getID() || gAgent.isGodlike()) { flags |= ESTATE_ACCESS_APPLY_TO_ALL_ESTATES; } else if (region->isEstateManager()) { flags |= ESTATE_ACCESS_APPLY_TO_MANAGED_ESTATES; } } LLFloaterRegionInfo::nextInvoice(); LLPanelEstateAccess::sendEstateAccessDelta(flags, itAgent->asUUID()); } } return false; } // Estate ban user // Derender //static void LLAvatarActions::derender(const LLUUID& agent_id, bool permanent) { LLAvatarNameCache::get(agent_id, boost::bind(&LLAvatarActions::onDerenderAvatarNameLookup, _1, _2, permanent)); } //static void LLAvatarActions::derenderMultiple(const uuid_vec_t& agent_ids, bool permanent) { for (uuid_vec_t::const_iterator it = agent_ids.begin(); it != agent_ids.end(); it++) { LLAvatarNameCache::get((*it), boost::bind(&LLAvatarActions::onDerenderAvatarNameLookup, _1, _2, permanent)); } } //static void LLAvatarActions::onDerenderAvatarNameLookup(const LLUUID& agent_id, const LLAvatarName& av_name, bool permanent) { FSAssetBlacklist::getInstance()->addNewItemToBlacklist(agent_id, av_name.getUserName(), "", LLAssetType::AT_PERSON, FSAssetBlacklist::eBlacklistFlag::NONE, permanent, permanent); LLViewerObject* av_obj = gObjectList.findObject(agent_id); if (av_obj) { gObjectList.killObject(av_obj); } } // Derender