/** * @file llpanelprofile.cpp * @brief Profile panel implementation * * $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 "llpanelprofile.h" #include "llagent.h" #include "llavataractions.h" #include "llfloaterreg.h" #include "llcommandhandler.h" #include "llpanelpicks.h" #include "lltabcontainer.h" static const std::string PANEL_PICKS = "panel_picks"; static const std::string PANEL_PROFILE = "panel_profile"; class LLAgentHandler : public LLCommandHandler { public: // requires trusted browser to trigger LLAgentHandler() : LLCommandHandler("agent", UNTRUSTED_THROTTLE) { } bool handle(const LLSD& params, const LLSD& query_map, LLMediaCtrl* web) { if (params.size() < 2) return false; LLUUID avatar_id; if (!avatar_id.set(params[0], FALSE)) { return false; } const std::string verb = params[1].asString(); if (verb == "about") { LLAvatarActions::showProfile(avatar_id); return true; } if (verb == "inspect") { LLFloaterReg::showInstance("inspect_avatar", LLSD().with("avatar_id", avatar_id)); return true; } if (verb == "im") { LLAvatarActions::startIM(avatar_id); return true; } if (verb == "pay") { LLAvatarActions::pay(avatar_id); return true; } if (verb == "offerteleport") { LLAvatarActions::offerTeleport(avatar_id); return true; } if (verb == "requestfriend") { LLAvatarActions::requestFriendshipDialog(avatar_id); return true; } if (verb == "mute") { if (! LLAvatarActions::isBlocked(avatar_id)) { LLAvatarActions::toggleBlock(avatar_id); } return true; } if (verb == "unmute") { if (LLAvatarActions::isBlocked(avatar_id)) { LLAvatarActions::toggleBlock(avatar_id); } return true; } return false; } }; LLAgentHandler gAgentHandler; //-- LLPanelProfile::ChildStack begins ---------------------------------------- LLPanelProfile::ChildStack::ChildStack(LLPanel* parent) : mParent(parent) { llassert_always(parent != NULL); } /// Save current parent's child views and remove them from the child list. bool LLPanelProfile::ChildStack::push() { view_list_t vlist = *mParent->getChildList(); for (view_list_t::const_iterator it = vlist.begin(); it != vlist.end(); ++it) { LLView* viewp = *it; mParent->removeChild(viewp); } mStack.push_back(vlist); dump(); return true; } /// Restore saved children (adding them back to the child list). bool LLPanelProfile::ChildStack::pop() { if (mStack.size() == 0) { llwarns << "Empty stack" << llendl; llassert(mStack.size() == 0); return false; } view_list_t& top = mStack.back(); for (view_list_t::const_iterator it = top.begin(); it != top.end(); ++it) { LLView* viewp = *it; mParent->addChild(viewp); } mStack.pop_back(); dump(); return true; } /// Temporarily add all saved children back. void LLPanelProfile::ChildStack::preParentReshape() { mSavedStack = mStack; while(mStack.size() > 0) { pop(); } } /// Add the temporarily saved children back. void LLPanelProfile::ChildStack::postParentReshape() { mStack = mSavedStack; mSavedStack = stack_t(); for (stack_t::const_iterator stack_it = mStack.begin(); stack_it != mStack.end(); ++stack_it) { const view_list_t& vlist = (*stack_it); for (view_list_t::const_iterator list_it = vlist.begin(); list_it != vlist.end(); ++list_it) { LLView* viewp = *list_it; lldebugs << "removing " << viewp->getName() << llendl; mParent->removeChild(viewp); } } } void LLPanelProfile::ChildStack::dump() { unsigned lvl = 0; lldebugs << "child stack dump:" << llendl; for (stack_t::const_iterator stack_it = mStack.begin(); stack_it != mStack.end(); ++stack_it, ++lvl) { std::ostringstream dbg_line; dbg_line << "lvl #" << lvl << ":"; const view_list_t& vlist = (*stack_it); for (view_list_t::const_iterator list_it = vlist.begin(); list_it != vlist.end(); ++list_it) { dbg_line << " " << (*list_it)->getName(); } lldebugs << dbg_line.str() << llendl; } } //-- LLPanelProfile::ChildStack ends ------------------------------------------ LLPanelProfile::LLPanelProfile() : LLPanel() , mTabCtrl(NULL) , mChildStack(this) , mAvatarId(LLUUID::null) { } BOOL LLPanelProfile::postBuild() { mTabCtrl = getChild("tabs"); getTabCtrl()->setCommitCallback(boost::bind(&LLPanelProfile::onTabSelected, this, _2)); LLPanelPicks* panel_picks = findChild(PANEL_PICKS); panel_picks->setProfilePanel(this); getTabContainer()[PANEL_PICKS] = panel_picks; getTabContainer()[PANEL_PROFILE] = findChild(PANEL_PROFILE); return TRUE; } // virtual void LLPanelProfile::reshape(S32 width, S32 height, BOOL called_from_parent) { // Temporarily add saved children back and reshape them. mChildStack.preParentReshape(); LLPanel::reshape(width, height, called_from_parent); mChildStack.postParentReshape(); } void LLPanelProfile::onOpen(const LLSD& key) { // open the desired panel if (key.has("open_tab_name")) { getTabContainer()[PANEL_PICKS]->onClosePanel(); // onOpen from selected panel will be called from onTabSelected callback getTabCtrl()->selectTabByName(key["open_tab_name"]); } else { getTabCtrl()->getCurrentPanel()->onOpen(getAvatarId()); } // support commands to open further pieces of UI if (key.has("show_tab_panel")) { std::string panel = key["show_tab_panel"].asString(); if (panel == "create_classified") { LLPanelPicks* picks = dynamic_cast(getTabContainer()[PANEL_PICKS]); if (picks) { picks->createNewClassified(); } } else if (panel == "classified_details") { LLPanelPicks* picks = dynamic_cast(getTabContainer()[PANEL_PICKS]); if (picks) { LLSD params = key; params.erase("show_tab_panel"); params.erase("open_tab_name"); picks->openClassifiedInfo(params); } } } } void LLPanelProfile::togglePanel(LLPanel* panel, const LLSD& key) { // TRUE - we need to open/expand "panel" bool expand = getChildList()->front() != panel; // mTabCtrl->getVisible(); if (expand) { openPanel(panel, key); } else { closePanel(panel); getTabCtrl()->getCurrentPanel()->onOpen(getAvatarId()); } } void LLPanelProfile::onTabSelected(const LLSD& param) { std::string tab_name = param.asString(); if (NULL != getTabContainer()[tab_name]) { getTabContainer()[tab_name]->onOpen(getAvatarId()); } } void LLPanelProfile::openPanel(LLPanel* panel, const LLSD& params) { // Hide currently visible panel (STORM-690). mChildStack.push(); // Add the panel or bring it to front. if (panel->getParent() != this) { addChild(panel); } else { sendChildToFront(panel); } panel->setVisible(TRUE); panel->setFocus(TRUE); // prevent losing focus by the floater panel->onOpen(params); LLRect new_rect = getRect(); panel->reshape(new_rect.getWidth(), new_rect.getHeight()); new_rect.setLeftTopAndSize(0, new_rect.getHeight(), new_rect.getWidth(), new_rect.getHeight()); panel->setRect(new_rect); } void LLPanelProfile::closePanel(LLPanel* panel) { panel->setVisible(FALSE); if (panel->getParent() == this) { removeChild(panel); // Make the underlying panel visible. mChildStack.pop(); // Prevent losing focus by the floater const child_list_t* child_list = getChildList(); if (child_list->size() > 0) { child_list->front()->setFocus(TRUE); } else { llwarns << "No underlying panel to focus." << llendl; } } } S32 LLPanelProfile::notifyParent(const LLSD& info) { std::string action = info["action"]; // lets update Picks list after Pick was saved if("save_new_pick" == action) { onOpen(info); return 1; } return LLPanel::notifyParent(info); }