1217 lines
37 KiB
C++
1217 lines
37 KiB
C++
/**
|
|
* @file llpanellogin.cpp
|
|
* @brief Login dialog and logo display
|
|
*
|
|
* $LicenseInfo:firstyear=2002&license=viewergpl$
|
|
*
|
|
* Copyright (c) 2002-2009, Linden Research, Inc.
|
|
*
|
|
* Second Life Viewer Source Code
|
|
* The source code in this file ("Source Code") is provided by Linden Lab
|
|
* to you under the terms of the GNU General Public License, version 2.0
|
|
* ("GPL"), unless you have obtained a separate licensing agreement
|
|
* ("Other License"), formally executed by you and Linden Lab. Terms of
|
|
* the GPL can be found in doc/GPL-license.txt in this distribution, or
|
|
* online at http://secondlifegrid.net/programs/open_source/licensing/gplv2
|
|
*
|
|
* There are special exceptions to the terms and conditions of the GPL as
|
|
* it is applied to this Source Code. View the full text of the exception
|
|
* in the file doc/FLOSS-exception.txt in this software distribution, or
|
|
* online at
|
|
* http://secondlifegrid.net/programs/open_source/licensing/flossexception
|
|
*
|
|
* By copying, modifying or distributing this software, you acknowledge
|
|
* that you have read and understood your obligations described above,
|
|
* and agree to abide by those obligations.
|
|
*
|
|
* ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO
|
|
* WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY,
|
|
* COMPLETENESS OR PERFORMANCE.
|
|
* $/LicenseInfo$
|
|
*/
|
|
|
|
#include "llviewerprecompiledheaders.h"
|
|
|
|
#include "llpanellogin.h"
|
|
|
|
#include "llpanelgeneral.h"
|
|
|
|
#include "hippogridmanager.h"
|
|
|
|
#include "indra_constants.h" // for key and mask constants
|
|
#include "llfontgl.h"
|
|
#include "llmd5.h"
|
|
#include "llsecondlifeurls.h"
|
|
#include "v4color.h"
|
|
|
|
#include "llappviewer.h"
|
|
#include "llbutton.h"
|
|
#include "llcheckboxctrl.h"
|
|
#include "llcommandhandler.h" // for secondlife:///app/login/
|
|
#include "llcombobox.h"
|
|
#include "llviewercontrol.h"
|
|
#include "llfloaterabout.h"
|
|
#include "llfloatertest.h"
|
|
#include "llfloaterpreference.h"
|
|
#include "llfocusmgr.h"
|
|
#include "lllineeditor.h"
|
|
#include "llnotificationsutil.h"
|
|
#include "llstartup.h"
|
|
#include "lltextbox.h"
|
|
#include "llui.h"
|
|
#include "lluiconstants.h"
|
|
#include "llurlhistory.h" // OGPX : regionuri text box has a history of region uris (if FN/LN are loaded at startup)
|
|
#include "llversioninfo.h"
|
|
#include "llviewertexturelist.h"
|
|
#include "llviewermenu.h" // for handle_preferences()
|
|
#include "llviewernetwork.h"
|
|
#include "llviewerwindow.h" // to link into child list
|
|
#include "llnotify.h"
|
|
#include "lluictrlfactory.h"
|
|
#include "llhttpclient.h"
|
|
#include "llweb.h"
|
|
#include "llmediactrl.h"
|
|
|
|
#include "llfloatertos.h"
|
|
|
|
#include "llglheaders.h"
|
|
|
|
// [RLVa:KB]
|
|
#include "rlvhandler.h"
|
|
// [/RLVa:KB]
|
|
|
|
// <edit>
|
|
#include "llspinctrl.h"
|
|
#include "llviewermessage.h"
|
|
#include <boost/lexical_cast.hpp>
|
|
// </edit>
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
#include "llsdserialize.h"
|
|
#include "llstring.h"
|
|
#include <cctype>
|
|
|
|
const S32 BLACK_BORDER_HEIGHT = 160;
|
|
const S32 MAX_PASSWORD = 16;
|
|
|
|
LLPanelLogin* LLPanelLogin::sInstance = NULL;
|
|
|
|
|
|
static bool nameSplit(const std::string& full, std::string& first, std::string& last)
|
|
{
|
|
std::vector<std::string> fragments;
|
|
boost::algorithm::split(fragments, full, boost::is_any_of(" ."));
|
|
if (!fragments.size() || !fragments[0].length())
|
|
return false;
|
|
first = fragments[0];
|
|
last = (fragments.size() == 1) ?
|
|
gHippoGridManager->getCurrentGrid()->isWhiteCore() ? LLStringUtil::null : "Resident" :
|
|
fragments[1];
|
|
return (fragments.size() <= 2);
|
|
}
|
|
|
|
static std::string nameJoin(const std::string& first,const std::string& last, bool strip_resident)
|
|
{
|
|
if (last.empty() || (strip_resident && boost::algorithm::iequals(last, "Resident")))
|
|
return first;
|
|
else if (std::islower(last[0]))
|
|
return first + '.' + last;
|
|
else
|
|
return first + ' ' + last;
|
|
}
|
|
|
|
static std::string getDisplayString(const std::string& first, const std::string& last, const std::string& grid, bool is_secondlife)
|
|
{
|
|
//grid comes via LLSavedLoginEntry, which uses full grid names, not nicks
|
|
if (grid == gHippoGridManager->getDefaultGridName())
|
|
return nameJoin(first, last, is_secondlife);
|
|
else
|
|
return nameJoin(first, last, is_secondlife) + " (" + grid + ')';
|
|
}
|
|
|
|
static std::string getDisplayString(const LLSavedLoginEntry& entry)
|
|
{
|
|
return getDisplayString(entry.getFirstName(), entry.getLastName(), entry.getGrid(), entry.isSecondLife());
|
|
}
|
|
|
|
|
|
class LLLoginLocationAutoHandler : public LLCommandHandler
|
|
{
|
|
public:
|
|
// don't allow from external browsers
|
|
LLLoginLocationAutoHandler() : LLCommandHandler("location_login", UNTRUSTED_BLOCK) { }
|
|
bool handle(const LLSD& tokens, const LLSD& query_map, LLMediaCtrl* web)
|
|
{
|
|
if (LLStartUp::getStartupState() < STATE_LOGIN_CLEANUP)
|
|
{
|
|
if ( tokens.size() == 0 || tokens.size() > 4 )
|
|
return false;
|
|
|
|
// unescape is important - uris with spaces are escaped in this code path
|
|
// (e.g. space -> %20) and the code to log into a region doesn't support that.
|
|
const std::string region = LLURI::unescape( tokens[0].asString() );
|
|
|
|
// just region name as payload
|
|
if ( tokens.size() == 1 )
|
|
{
|
|
// region name only - slurl will end up as center of region
|
|
LLSLURL slurl(region);
|
|
LLPanelLogin::autologinToLocation(slurl);
|
|
}
|
|
else
|
|
// region name and x coord as payload
|
|
if ( tokens.size() == 2 )
|
|
{
|
|
// invalid to only specify region and x coordinate
|
|
// slurl code will revert to same as region only, so do this anyway
|
|
LLSLURL slurl(region);
|
|
LLPanelLogin::autologinToLocation(slurl);
|
|
}
|
|
else
|
|
// region name and x/y coord as payload
|
|
if ( tokens.size() == 3 )
|
|
{
|
|
// region and x/y specified - default z to 0
|
|
F32 xpos;
|
|
std::istringstream codec(tokens[1].asString());
|
|
codec >> xpos;
|
|
|
|
F32 ypos;
|
|
codec.clear();
|
|
codec.str(tokens[2].asString());
|
|
codec >> ypos;
|
|
|
|
const LLVector3 location(xpos, ypos, 0.0f);
|
|
LLSLURL slurl(region, location);
|
|
|
|
LLPanelLogin::autologinToLocation(slurl);
|
|
}
|
|
else
|
|
// region name and x/y/z coord as payload
|
|
if ( tokens.size() == 4 )
|
|
{
|
|
// region and x/y/z specified - ok
|
|
F32 xpos;
|
|
std::istringstream codec(tokens[1].asString());
|
|
codec >> xpos;
|
|
|
|
F32 ypos;
|
|
codec.clear();
|
|
codec.str(tokens[2].asString());
|
|
codec >> ypos;
|
|
|
|
F32 zpos;
|
|
codec.clear();
|
|
codec.str(tokens[3].asString());
|
|
codec >> zpos;
|
|
|
|
const LLVector3 location(xpos, ypos, zpos);
|
|
LLSLURL slurl(region, location);
|
|
|
|
LLPanelLogin::autologinToLocation(slurl);
|
|
};
|
|
}
|
|
return true;
|
|
}
|
|
};
|
|
LLLoginLocationAutoHandler gLoginLocationAutoHandler;
|
|
|
|
//---------------------------------------------------------------------------
|
|
// Public methods
|
|
//---------------------------------------------------------------------------
|
|
LLPanelLogin::LLPanelLogin(const LLRect& rect)
|
|
: LLPanel(std::string("panel_login"), rect, FALSE), // not bordered
|
|
mLogoImage(LLUI::getUIImage("startup_logo.j2c"))
|
|
{
|
|
setFocusRoot(TRUE);
|
|
|
|
setBackgroundVisible(FALSE);
|
|
setBackgroundOpaque(TRUE);
|
|
|
|
LLPanelLogin::sInstance = this;
|
|
|
|
// add to front so we are the bottom-most child
|
|
gViewerWindow->getRootView()->addChildInBack(this);
|
|
|
|
LLUICtrlFactory::getInstance()->buildPanel(this, "panel_login.xml");
|
|
|
|
reshape(rect.getWidth(), rect.getHeight());
|
|
|
|
LLComboBox* username_combo(getChild<LLComboBox>("username_combo"));
|
|
username_combo->setCommitCallback(boost::bind(LLPanelLogin::onSelectLoginEntry, _2));
|
|
username_combo->setFocusLostCallback(boost::bind(&LLPanelLogin::onLoginComboLostFocus, this, username_combo));
|
|
username_combo->setPrevalidate(LLLineEditor::prevalidatePrintableNotPipe);
|
|
username_combo->setSuppressTentative(true);
|
|
username_combo->setSuppressAutoComplete(true);
|
|
|
|
getChild<LLUICtrl>("remember_name_check")->setCommitCallback(boost::bind(&LLPanelLogin::onNameCheckChanged, this, _2));
|
|
|
|
LLLineEditor* password_edit(getChild<LLLineEditor>("password_edit"));
|
|
password_edit->setKeystrokeCallback(boost::bind(LLPanelLogin::onPassKey));
|
|
// STEAM-14: When user presses Enter with this field in focus, initiate login
|
|
password_edit->setCommitCallback(boost::bind(&LLPanelLogin::mungePassword, this, _2));
|
|
password_edit->setDrawAsterixes(TRUE);
|
|
|
|
getChild<LLUICtrl>("remove_login")->setCommitCallback(boost::bind(&LLPanelLogin::confirmDelete, this));
|
|
|
|
// change z sort of clickable text to be behind buttons
|
|
sendChildToBack(getChildView("channel_text"));
|
|
sendChildToBack(getChildView("forgot_password_text"));
|
|
|
|
//LL_INFOS() << " url history: " << LLSDOStreamer<LLSDXMLFormatter>(LLURLHistory::getURLHistory("regionuri")) << LL_ENDL;
|
|
|
|
LLComboBox* location_combo = getChild<LLComboBox>("start_location_combo");
|
|
updateLocationSelectorsVisibility(); // separate so that it can be called from preferences
|
|
location_combo->setAllowTextEntry(TRUE, 128, FALSE);
|
|
location_combo->setFocusLostCallback( boost::bind(&LLPanelLogin::onLocationSLURL, this) );
|
|
|
|
LLComboBox* server_choice_combo = getChild<LLComboBox>("grids_combo");
|
|
server_choice_combo->setCommitCallback(boost::bind(&LLPanelLogin::onSelectGrid, _1));
|
|
server_choice_combo->setFocusLostCallback(boost::bind(&LLPanelLogin::onSelectGrid, server_choice_combo));
|
|
|
|
// Load all of the grids, sorted, and then add a bar and the current grid at the top
|
|
updateGridCombo();
|
|
|
|
LLSLURL start_slurl(LLStartUp::getStartSLURL());
|
|
if (!start_slurl.isSpatial()) // has a start been established by the command line or NextLoginLocation ?
|
|
{
|
|
// no, so get the preference setting
|
|
std::string defaultStartLocation = gSavedSettings.getString("LoginLocation");
|
|
LL_INFOS("AppInit")<<"default LoginLocation '" << defaultStartLocation << '\'' << LL_ENDL;
|
|
LLSLURL defaultStart(defaultStartLocation);
|
|
if ( defaultStart.isSpatial() )
|
|
{
|
|
LLStartUp::setStartSLURL(defaultStart); // calls onUpdateStartSLURL
|
|
}
|
|
else
|
|
{
|
|
LL_INFOS("AppInit")<<"no valid LoginLocation, using home"<<LL_ENDL;
|
|
LLSLURL homeStart(LLSLURL::SIM_LOCATION_HOME);
|
|
LLStartUp::setStartSLURL(homeStart); // calls onUpdateStartSLURL
|
|
}
|
|
}
|
|
else
|
|
{
|
|
LLPanelLogin::onUpdateStartSLURL(start_slurl); // updates grid if needed
|
|
}
|
|
|
|
|
|
// Also set default button for subpanels, otherwise hitting enter in text entry fields won't login
|
|
{
|
|
LLButton* connect_btn(findChild<LLButton>("connect_btn"));
|
|
connect_btn->setCommitCallback(boost::bind(&LLPanelLogin::onClickConnect, this));
|
|
setDefaultBtn(connect_btn);
|
|
findChild<LLPanel>("name_panel")->setDefaultBtn(connect_btn);
|
|
findChild<LLPanel>("password_panel")->setDefaultBtn(connect_btn);
|
|
findChild<LLPanel>("grids_panel")->setDefaultBtn(connect_btn);
|
|
findChild<LLPanel>("location_panel")->setDefaultBtn(connect_btn);
|
|
findChild<LLPanel>("login_html")->setDefaultBtn(connect_btn);
|
|
}
|
|
|
|
getChild<LLUICtrl>("grids_btn")->setCommitCallback(boost::bind(LLPanelLogin::onClickGrids));
|
|
|
|
std::string channel = LLVersionInfo::getChannel();
|
|
|
|
std::string version = llformat("%s (%d)",
|
|
LLVersionInfo::getShortVersion().c_str(),
|
|
LLVersionInfo::getBuild());
|
|
|
|
LLTextBox* channel_text = getChild<LLTextBox>("channel_text");
|
|
channel_text->setTextArg("[CHANNEL]", channel); // though not displayed
|
|
channel_text->setTextArg("[VERSION]", version);
|
|
channel_text->setClickedCallback(boost::bind(LLFloaterAbout::show,(void*)NULL));
|
|
|
|
LLTextBox* forgot_password_text = getChild<LLTextBox>("forgot_password_text");
|
|
forgot_password_text->setClickedCallback(boost::bind(&onClickForgotPassword));
|
|
|
|
|
|
LLTextBox* create_new_account_text = getChild<LLTextBox>("create_new_account_text");
|
|
create_new_account_text->setClickedCallback(boost::bind(&onClickNewAccount));
|
|
|
|
// get the web browser control
|
|
LLMediaCtrl* web_browser = getChild<LLMediaCtrl>("login_html");
|
|
web_browser->addObserver(this);
|
|
web_browser->setBackgroundColor(LLColor4::black);
|
|
|
|
reshapeBrowser();
|
|
|
|
refreshLoginPage();
|
|
|
|
gHippoGridManager->setCurrentGridChangeCallback(boost::bind(&LLPanelLogin::onCurGridChange,this,_1,_2));
|
|
|
|
// Load login history
|
|
std::string login_hist_filepath = gDirUtilp->getExpandedFilename(LL_PATH_USER_SETTINGS, "saved_logins_sg2.xml");
|
|
mLoginHistoryData = LLSavedLogins::loadFile(login_hist_filepath);
|
|
|
|
const LLSavedLoginsList& saved_login_entries(mLoginHistoryData.getEntries());
|
|
for (LLSavedLoginsList::const_reverse_iterator i = saved_login_entries.rbegin();
|
|
i != saved_login_entries.rend(); ++i)
|
|
{
|
|
const LLSD& e = i->asLLSD();
|
|
if (e.isMap() && gHippoGridManager->getGrid(i->getGrid()))
|
|
username_combo->add(getDisplayString(*i), e);
|
|
}
|
|
|
|
if (saved_login_entries.size() > 0)
|
|
{
|
|
setFields(*saved_login_entries.rbegin());
|
|
}
|
|
|
|
addFavoritesToStartLocation();
|
|
}
|
|
|
|
void LLPanelLogin::addFavoritesToStartLocation()
|
|
{
|
|
// Clear the combo.
|
|
auto combo = getChild<LLComboBox>("start_location_combo");
|
|
if (!combo) return;
|
|
S32 num_items = combo->getItemCount();
|
|
for (S32 i = num_items - 1; i > 2; i--)
|
|
{
|
|
combo->remove(i);
|
|
}
|
|
|
|
// Load favorites into the combo.
|
|
const auto grid = gHippoGridManager->getCurrentGrid();
|
|
std::string first, last, password;
|
|
getFields(first, last, password);
|
|
auto user_defined_name(first + ' ' + last);
|
|
std::string filename = gDirUtilp->getExpandedFilename(LL_PATH_USER_SETTINGS, "stored_favorites_" + grid->getGridName() + ".xml");
|
|
std::string old_filename = gDirUtilp->getExpandedFilename(LL_PATH_USER_SETTINGS, "stored_favorites.xml");
|
|
|
|
LLSD fav_llsd;
|
|
llifstream file;
|
|
file.open(filename);
|
|
if (!file.is_open())
|
|
{
|
|
file.open(old_filename);
|
|
if (!file.is_open()) return;
|
|
}
|
|
LLSDSerialize::fromXML(fav_llsd, file);
|
|
|
|
for (LLSD::map_const_iterator iter = fav_llsd.beginMap();
|
|
iter != fav_llsd.endMap(); ++iter)
|
|
{
|
|
// The account name in stored_favorites.xml has Resident last name even if user has
|
|
// a single word account name, so it can be compared case-insensitive with the
|
|
// user defined "firstname lastname".
|
|
S32 res = LLStringUtil::compareInsensitive(user_defined_name, iter->first);
|
|
if (res != 0)
|
|
{
|
|
LL_DEBUGS() << "Skipping favorites for " << iter->first << LL_ENDL;
|
|
continue;
|
|
}
|
|
|
|
combo->addSeparator();
|
|
LL_DEBUGS() << "Loading favorites for " << iter->first << LL_ENDL;
|
|
auto user_llsd = iter->second;
|
|
for (LLSD::array_const_iterator iter1 = user_llsd.beginArray();
|
|
iter1 != user_llsd.endArray(); ++iter1)
|
|
{
|
|
std::string label = (*iter1)["name"].asString();
|
|
std::string value = (*iter1)["slurl"].asString();
|
|
if (!label.empty() && !value.empty())
|
|
{
|
|
combo->add(label, value);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
void LLPanelLogin::setSiteIsAlive(bool alive)
|
|
{
|
|
if (LLMediaCtrl* web_browser = getChild<LLMediaCtrl>("login_html"))
|
|
{
|
|
if (alive) // if the contents of the site was retrieved
|
|
loadLoginPage();
|
|
else // the site is not available (missing page, server down, other badness)
|
|
web_browser->navigateTo( "data:text/html,%3Chtml%3E%3Cbody%20bgcolor=%22#000000%22%3E%3C/body%3E%3C/html%3E", "text/html" );
|
|
web_browser->setVisible(alive);
|
|
}
|
|
}
|
|
|
|
void LLPanelLogin::clearPassword()
|
|
{
|
|
getChild<LLUICtrl>("password_edit")->setValue(mIncomingPassword = mMungedPassword = LLStringUtil::null);
|
|
}
|
|
|
|
void LLPanelLogin::hidePassword()
|
|
{
|
|
// This is a MD5 hex digest of a password.
|
|
// We don't actually use the password input field,
|
|
// fill it with MAX_PASSWORD characters so we get a
|
|
// nice row of asterixes.
|
|
getChild<LLUICtrl>("password_edit")->setValue("123456789!123456");
|
|
}
|
|
|
|
void LLPanelLogin::mungePassword(const std::string& password)
|
|
{
|
|
// Re-md5 if we've changed at all
|
|
if (password != mIncomingPassword)
|
|
{
|
|
// Max "actual" password length is 16 characters.
|
|
// Hex digests are always 32 characters.
|
|
if (password.length() == MD5HEX_STR_BYTES)
|
|
{
|
|
hidePassword();
|
|
mMungedPassword = password;
|
|
}
|
|
else
|
|
{
|
|
LLMD5 pass((unsigned char *)utf8str_truncate(password, gHippoGridManager->getCurrentGrid()->isOpenSimulator() ? 24 : 16).c_str());
|
|
char munged_password[MD5HEX_STR_SIZE];
|
|
pass.hex_digest(munged_password);
|
|
mMungedPassword = munged_password;
|
|
}
|
|
mIncomingPassword = password;
|
|
}
|
|
}
|
|
|
|
// force the size to be correct (XML doesn't seem to be sufficient to do this)
|
|
// (with some padding so the other login screen doesn't show through)
|
|
void LLPanelLogin::reshapeBrowser()
|
|
{
|
|
auto web_browser = getChild<LLMediaCtrl>("login_html");
|
|
LLRect rect = gViewerWindow->getWindowRectScaled();
|
|
LLRect html_rect;
|
|
html_rect.setCenterAndSize(
|
|
rect.getCenterX() /*- 2*/, rect.getCenterY() + 40,
|
|
rect.getWidth() /*+ 6*/, rect.getHeight() - 78 );
|
|
web_browser->setRect( html_rect );
|
|
web_browser->reshape( html_rect.getWidth(), html_rect.getHeight(), TRUE );
|
|
reshape( rect.getWidth(), rect.getHeight(), 1 );
|
|
}
|
|
|
|
LLPanelLogin::~LLPanelLogin()
|
|
{
|
|
std::string login_hist_filepath = gDirUtilp->getExpandedFilename(LL_PATH_USER_SETTINGS, "saved_logins_sg2.xml");
|
|
LLSavedLogins::saveFile(mLoginHistoryData, login_hist_filepath);
|
|
|
|
sInstance = nullptr;
|
|
|
|
// Controls having keyboard focus by default
|
|
// must reset it on destroy. (EXT-2748)
|
|
if (gFocusMgr.getDefaultKeyboardFocus() == this)
|
|
gFocusMgr.setDefaultKeyboardFocus(nullptr);
|
|
}
|
|
|
|
// virtual
|
|
void LLPanelLogin::draw()
|
|
{
|
|
gGL.pushMatrix();
|
|
{
|
|
F32 image_aspect = 1.333333f;
|
|
F32 view_aspect = (F32)getRect().getWidth() / (F32)getRect().getHeight();
|
|
// stretch image to maintain aspect ratio
|
|
if (image_aspect > view_aspect)
|
|
{
|
|
gGL.translatef(-0.5f * (image_aspect / view_aspect - 1.f) * getRect().getWidth(), 0.f, 0.f);
|
|
gGL.scalef(image_aspect / view_aspect, 1.f, 1.f);
|
|
}
|
|
|
|
S32 width = getRect().getWidth();
|
|
S32 height = getRect().getHeight();
|
|
|
|
if ( getChild<LLView>("login_html")->getVisible())
|
|
{
|
|
// draw a background box in black
|
|
gl_rect_2d( 0, height - 264, width, 264, LLColor4::black );
|
|
// draw the bottom part of the background image
|
|
// just the blue background to the native client UI
|
|
mLogoImage->draw(0, -264, width + 8, mLogoImage->getHeight());
|
|
}
|
|
else
|
|
{
|
|
// the HTML login page is not available so default to the original screen
|
|
S32 offscreen_part = height / 3;
|
|
mLogoImage->draw(0, -offscreen_part, width, height+offscreen_part);
|
|
};
|
|
}
|
|
gGL.popMatrix();
|
|
|
|
LLPanel::draw();
|
|
}
|
|
|
|
// virtual
|
|
BOOL LLPanelLogin::handleKeyHere(KEY key, MASK mask)
|
|
{
|
|
if (('T' == key) && (MASK_CONTROL == mask))
|
|
{
|
|
new LLFloaterSimple("floater_test.xml");
|
|
return TRUE;
|
|
}
|
|
|
|
# if !LL_RELEASE_FOR_DOWNLOAD
|
|
if ( KEY_F2 == key )
|
|
{
|
|
LL_INFOS() << "Spawning floater TOS window" << LL_ENDL;
|
|
LLFloaterTOS* tos_dialog = LLFloaterTOS::show(LLFloaterTOS::TOS_TOS,LLStringUtil::null);
|
|
tos_dialog->startModal();
|
|
return TRUE;
|
|
}
|
|
#endif
|
|
|
|
return LLPanel::handleKeyHere(key, mask);
|
|
}
|
|
|
|
// virtual
|
|
void LLPanelLogin::setFocus(BOOL b)
|
|
{
|
|
if(b != hasFocus())
|
|
{
|
|
if(b)
|
|
{
|
|
LLPanelLogin::giveFocus();
|
|
}
|
|
else
|
|
{
|
|
LLPanel::setFocus(b);
|
|
}
|
|
}
|
|
}
|
|
|
|
// static
|
|
void LLPanelLogin::giveFocus()
|
|
{
|
|
if( sInstance )
|
|
{
|
|
if (!sInstance->getVisible()) sInstance->setVisible(true);
|
|
// Grab focus and move cursor to first blank input field
|
|
std::string username = sInstance->getChild<LLUICtrl>("username_combo")->getValue().asString();
|
|
std::string pass = sInstance->getChild<LLUICtrl>("password_edit")->getValue().asString();
|
|
|
|
BOOL have_username = !username.empty();
|
|
BOOL have_pass = !pass.empty();
|
|
|
|
LLLineEditor* edit = nullptr;
|
|
LLComboBox* combo = nullptr;
|
|
if (have_username && !have_pass)
|
|
{
|
|
// User saved his name but not his password. Move
|
|
// focus to password field.
|
|
edit = sInstance->getChild<LLLineEditor>("password_edit");
|
|
}
|
|
else
|
|
{
|
|
// User doesn't have a name, so start there.
|
|
combo = sInstance->getChild<LLComboBox>("username_combo");
|
|
}
|
|
|
|
if (edit)
|
|
{
|
|
edit->setFocus(TRUE);
|
|
edit->selectAll();
|
|
}
|
|
else if (combo)
|
|
{
|
|
combo->setFocus(TRUE);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// static
|
|
void LLPanelLogin::show()
|
|
{
|
|
if (sInstance) sInstance->setVisible(true);
|
|
else new LLPanelLogin(gViewerWindow->getVirtualWindowRect());
|
|
|
|
if( !gFocusMgr.getKeyboardFocus() )
|
|
{
|
|
// Grab focus and move cursor to first enabled control
|
|
sInstance->setFocus(TRUE);
|
|
}
|
|
|
|
// Make sure that focus always goes here (and use the latest sInstance that was just created)
|
|
gFocusMgr.setDefaultKeyboardFocus(sInstance);
|
|
}
|
|
|
|
// static
|
|
void LLPanelLogin::setFields(const std::string& firstname,
|
|
const std::string& lastname,
|
|
const std::string& password)
|
|
{
|
|
if (!sInstance)
|
|
{
|
|
LL_WARNS() << "Attempted fillFields with no login view shown" << LL_ENDL;
|
|
return;
|
|
}
|
|
|
|
LLComboBox* login_combo = sInstance->getChild<LLComboBox>("username_combo");
|
|
|
|
llassert_always(firstname.find(' ') == std::string::npos);
|
|
login_combo->setLabel(nameJoin(firstname, lastname, false));
|
|
|
|
sInstance->mungePassword(password);
|
|
if (sInstance->mIncomingPassword != sInstance->mMungedPassword)
|
|
sInstance->getChild<LLUICtrl>("password_edit")->setValue(password);
|
|
else
|
|
sInstance->hidePassword();
|
|
}
|
|
|
|
// static
|
|
void LLPanelLogin::setFields(const LLSavedLoginEntry& entry, bool takeFocus)
|
|
{
|
|
if (!sInstance)
|
|
{
|
|
LL_WARNS() << "Attempted setFields with no login view shown" << LL_ENDL;
|
|
return;
|
|
}
|
|
|
|
LLCheckBoxCtrl* remember_pass_check = sInstance->getChild<LLCheckBoxCtrl>("remember_check");
|
|
std::string fullname = nameJoin(entry.getFirstName(), entry.getLastName(), entry.isSecondLife());
|
|
LLComboBox* login_combo = sInstance->getChild<LLComboBox>("username_combo");
|
|
login_combo->setTextEntry(fullname);
|
|
login_combo->resetTextDirty();
|
|
//login_combo->setValue(fullname);
|
|
|
|
const auto& grid = entry.getGrid();
|
|
//grid comes via LLSavedLoginEntry, which uses full grid names, not nicks
|
|
if (!grid.empty() && gHippoGridManager->getGrid(grid) && grid != gHippoGridManager->getCurrentGridName())
|
|
{
|
|
gHippoGridManager->setCurrentGrid(grid);
|
|
}
|
|
|
|
const auto& password = entry.getPassword();
|
|
bool remember_pass = !password.empty();
|
|
if (remember_pass)
|
|
{
|
|
sInstance->mIncomingPassword = sInstance->mMungedPassword = password;
|
|
sInstance->hidePassword();
|
|
}
|
|
else sInstance->clearPassword();
|
|
remember_pass_check->setValue(remember_pass);
|
|
|
|
if (takeFocus) giveFocus();
|
|
}
|
|
|
|
// static
|
|
void LLPanelLogin::getFields(std::string& firstname, std::string& lastname, std::string& password)
|
|
{
|
|
if (!sInstance)
|
|
{
|
|
LL_WARNS() << "Attempted getFields with no login view shown" << LL_ENDL;
|
|
return;
|
|
}
|
|
|
|
nameSplit(sInstance->getChild<LLComboBox>("username_combo")->getTextEntry(), firstname, lastname);
|
|
LLStringUtil::trim(firstname);
|
|
LLStringUtil::trim(lastname);
|
|
|
|
password = sInstance->mMungedPassword;
|
|
}
|
|
|
|
// static
|
|
/*void LLPanelLogin::getLocation(std::string &location)
|
|
{
|
|
if (!sInstance)
|
|
{
|
|
LL_WARNS() << "Attempted getLocation with no login view shown" << LL_ENDL;
|
|
return;
|
|
}
|
|
|
|
LLComboBox* combo = sInstance->getChild<LLComboBox>("start_location_combo");
|
|
location = combo->getValue().asString();
|
|
}*/
|
|
|
|
// static
|
|
void LLPanelLogin::updateLocationSelectorsVisibility()
|
|
{
|
|
if (sInstance)
|
|
{
|
|
BOOL show_start = gSavedSettings.getBOOL("ShowStartLocation");
|
|
|
|
// [RLVa:KB] - Alternate: Snowglobe-1.2.4 | Checked: 2009-07-08 (RLVa-1.0.0e)
|
|
// TODO-RLVa: figure out some way to make this work with RLV_EXTENSION_STARTLOCATION
|
|
#ifndef RLV_EXTENSION_STARTLOCATION
|
|
if (rlv_handler_t::isEnabled())
|
|
{
|
|
show_start = FALSE;
|
|
}
|
|
#endif // RLV_EXTENSION_STARTLOCATION
|
|
// [/RLVa:KB]
|
|
|
|
sInstance->getChildView("location_panel")->setVisible(show_start);
|
|
}
|
|
|
|
}
|
|
|
|
// static
|
|
void LLPanelLogin::onUpdateStartSLURL(const LLSLURL& new_start_slurl)
|
|
{
|
|
if (!sInstance) return;
|
|
|
|
LL_DEBUGS("AppInit")<<new_start_slurl.asString()<<LL_ENDL;
|
|
|
|
auto location_combo = sInstance->getChild<LLComboBox>("start_location_combo");
|
|
/*
|
|
* Determine whether or not the new_start_slurl modifies the grid.
|
|
*
|
|
* Note that some forms that could be in the slurl are grid-agnostic.,
|
|
* such as "home". Other forms, such as
|
|
* https://grid.example.com/region/Party%20Town/20/30/5
|
|
* specify a particular grid; in those cases we want to change the grid
|
|
* and the grid selector to match the new value.
|
|
*/
|
|
enum LLSLURL::SLURL_TYPE new_slurl_type = new_start_slurl.getType();
|
|
switch (new_slurl_type)
|
|
{
|
|
case LLSLURL::LOCATION:
|
|
{
|
|
location_combo->setCurrentByIndex(2);
|
|
location_combo->setTextEntry(new_start_slurl.getLocationString());
|
|
}
|
|
break;
|
|
|
|
case LLSLURL::HOME_LOCATION:
|
|
location_combo->setCurrentByIndex(0); // home location
|
|
break;
|
|
|
|
case LLSLURL::LAST_LOCATION:
|
|
location_combo->setCurrentByIndex(1); // last location
|
|
break;
|
|
|
|
default:
|
|
LL_WARNS("AppInit")<<"invalid login slurl, using home"<<LL_ENDL;
|
|
location_combo->setCurrentByIndex(1); // home location
|
|
break;
|
|
}
|
|
|
|
updateLocationSelectorsVisibility();
|
|
}
|
|
|
|
void LLPanelLogin::setLocation(const LLSLURL& slurl)
|
|
{
|
|
LL_DEBUGS("AppInit")<<"setting Location "<<slurl.asString()<<LL_ENDL;
|
|
LLStartUp::setStartSLURL(slurl); // calls onUpdateStartSLURL, above
|
|
}
|
|
|
|
void LLPanelLogin::autologinToLocation(const LLSLURL& slurl)
|
|
{
|
|
LL_DEBUGS("AppInit")<<"automatically logging into Location "<<slurl.asString()<<LL_ENDL;
|
|
LLStartUp::setStartSLURL(slurl); // calls onUpdateStartSLURL, above
|
|
|
|
if ( LLPanelLogin::sInstance != NULL )
|
|
{
|
|
LLPanelLogin::sInstance->onClickConnect();
|
|
}
|
|
}
|
|
|
|
|
|
// static
|
|
void LLPanelLogin::close()
|
|
{
|
|
if (sInstance)
|
|
{
|
|
sInstance->getParent()->removeChild(sInstance);
|
|
|
|
delete sInstance;
|
|
sInstance = nullptr;
|
|
}
|
|
}
|
|
|
|
// static
|
|
void LLPanelLogin::setAlwaysRefresh(bool refresh)
|
|
{
|
|
if (sInstance && LLStartUp::getStartupState() < STATE_LOGIN_CLEANUP)
|
|
if (LLMediaCtrl* web_browser = sInstance->getChild<LLMediaCtrl>("login_html"))
|
|
web_browser->setAlwaysRefresh(refresh);
|
|
}
|
|
|
|
void LLPanelLogin::updateGridCombo()
|
|
{
|
|
const std::string& defaultGrid = gHippoGridManager->getDefaultGridName();
|
|
|
|
LLComboBox* grids = getChild<LLComboBox>("grids_combo");
|
|
std::string top_entry;
|
|
|
|
grids->removeall();
|
|
|
|
const HippoGridInfo* curGrid = gHippoGridManager->getCurrentGrid();
|
|
const HippoGridInfo* defGrid = gHippoGridManager->getGrid(defaultGrid);
|
|
|
|
S32 idx(-1);
|
|
HippoGridManager::GridIterator it, end = gHippoGridManager->endGrid();
|
|
for (it = gHippoGridManager->beginGrid(); it != end; ++it)
|
|
{
|
|
std::string grid = it->second->getGridName();
|
|
if (grid.empty() || it->second == defGrid)
|
|
continue;
|
|
if (it->second == curGrid) idx = grids->getItemCount();
|
|
grids->add(grid);
|
|
}
|
|
if (curGrid || defGrid)
|
|
{
|
|
if (defGrid)
|
|
{
|
|
grids->add(defGrid->getGridName(), ADD_TOP);
|
|
++idx;
|
|
}
|
|
grids->setCurrentByIndex(idx);
|
|
}
|
|
else
|
|
{
|
|
grids->setLabel(LLStringUtil::null); // LLComboBox::removeall() does not clear the label
|
|
}
|
|
}
|
|
|
|
void LLPanelLogin::loadLoginPage()
|
|
{
|
|
if (!sInstance) return;
|
|
|
|
sInstance->updateGridCombo();
|
|
|
|
std::string login_page_str = gHippoGridManager->getCurrentGrid()->getLoginPage();
|
|
if (login_page_str.empty())
|
|
{
|
|
sInstance->setSiteIsAlive(false);
|
|
return;
|
|
}
|
|
|
|
// Use the right delimeter depending on how LLURI parses the URL
|
|
LLURI login_page = LLURI(login_page_str);
|
|
LLSD params(login_page.queryMap());
|
|
|
|
LL_DEBUGS("AppInit") << "login_page: " << login_page << LL_ENDL;
|
|
|
|
// Language
|
|
params["lang"] = LLUI::getLanguage();
|
|
|
|
// First Login?
|
|
if (gSavedSettings.getBOOL("FirstLoginThisInstall"))
|
|
{
|
|
params["firstlogin"] = "TRUE"; // not bool: server expects string TRUE
|
|
}
|
|
|
|
// Channel and Version
|
|
params["version"] = llformat("%s (%d)",
|
|
LLVersionInfo::getShortVersion().c_str(),
|
|
LLVersionInfo::getBuild());
|
|
params["channel"] = LLVersionInfo::getChannel();
|
|
|
|
// Grid
|
|
|
|
if (gHippoGridManager->getCurrentGrid()->isSecondLife())
|
|
{
|
|
// find second life grid from login URI
|
|
// yes, this is heuristic, but hey, it is just to get the right login page...
|
|
std::string tmp = gHippoGridManager->getCurrentGrid()->getLoginUri();
|
|
int i = tmp.find(".lindenlab.com");
|
|
if (i != std::string::npos) {
|
|
tmp = tmp.substr(0, i);
|
|
i = tmp.rfind('.');
|
|
if (i == std::string::npos)
|
|
i = tmp.rfind('/');
|
|
if (i != std::string::npos) {
|
|
tmp = tmp.substr(i+1);
|
|
params["grid"] = tmp;
|
|
}
|
|
}
|
|
}
|
|
else if (gHippoGridManager->getCurrentGrid()->isOpenSimulator())
|
|
{
|
|
params["grid"] = gHippoGridManager->getCurrentGrid()->getGridNick();
|
|
}
|
|
else if (gHippoGridManager->getCurrentGrid()->getPlatform() == HippoGridInfo::PLATFORM_WHITECORE)
|
|
{
|
|
params["grid"] = LLViewerLogin::getInstance()->getGridLabel();
|
|
}
|
|
|
|
// add OS info
|
|
params["os"] = LLAppViewer::instance()->getOSInfo().getOSStringSimple();
|
|
|
|
// Make an LLURI with this augmented info
|
|
LLURI login_uri(LLURI::buildHTTP(login_page.authority(),
|
|
login_page.path(),
|
|
params));
|
|
|
|
gViewerWindow->setMenuBackgroundColor(false, !LLViewerLogin::getInstance()->isInProductionGrid());
|
|
gLoginMenuBarView->setBackgroundColor(gMenuBarView->getBackgroundColor());
|
|
|
|
std::string singularity_splash_uri = gSavedSettings.getString("SingularitySplashPagePrefix");
|
|
if (!singularity_splash_uri.empty())
|
|
{
|
|
params["original_page"] = login_uri.asString();
|
|
login_uri = LLURI::buildHTTP(singularity_splash_uri, gSavedSettings.getString("SingularitySplashPagePath"), params);
|
|
}
|
|
|
|
LLMediaCtrl* web_browser = sInstance->getChild<LLMediaCtrl>("login_html");
|
|
if (web_browser->getCurrentNavUrl() != login_uri.asString())
|
|
{
|
|
LL_DEBUGS("AppInit") << "loading: " << login_uri << LL_ENDL;
|
|
web_browser->navigateTo( login_uri.asString(), "text/html" );
|
|
}
|
|
}
|
|
|
|
void LLPanelLogin::handleMediaEvent(LLPluginClassMedia* /*self*/, EMediaEvent event)
|
|
{
|
|
}
|
|
|
|
//---------------------------------------------------------------------------
|
|
// Protected methods
|
|
//---------------------------------------------------------------------------
|
|
|
|
void LLPanelLogin::onClickConnect()
|
|
{
|
|
// JC - Make sure the fields all get committed.
|
|
gFocusMgr.setKeyboardFocus(NULL);
|
|
|
|
std::string first, last;
|
|
if (nameSplit(getChild<LLComboBox>("username_combo")->getTextEntry(), first, last))
|
|
LLStartUp::setStartupState(STATE_LOGIN_CLEANUP);
|
|
else if (gHippoGridManager->getCurrentGrid()->getRegisterUrl().empty())
|
|
LLNotificationsUtil::add("MustHaveAccountToLogInNoLinks");
|
|
else
|
|
LLNotificationsUtil::add("MustHaveAccountToLogIn", LLSD(), LLSD(),
|
|
LLPanelLogin::newAccountAlertCallback);
|
|
}
|
|
|
|
|
|
// static
|
|
bool LLPanelLogin::newAccountAlertCallback(const LLSD& notification, const LLSD& response)
|
|
{
|
|
if (0 == LLNotification::getSelectedOption(notification, response))
|
|
{
|
|
LL_INFOS() << "Going to account creation URL" << LL_ENDL;
|
|
LLWeb::loadURLExternal(CREATE_ACCOUNT_URL);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
|
|
// static
|
|
void LLPanelLogin::onClickNewAccount()
|
|
{
|
|
const std::string& url = gHippoGridManager->getCurrentGrid()->getRegisterUrl();
|
|
if (!url.empty())
|
|
{
|
|
LL_INFOS() << "Going to account creation URL." << LL_ENDL;
|
|
LLWeb::loadURLExternal(url);
|
|
}
|
|
else
|
|
{
|
|
LL_INFOS() << "Account creation URL is empty." << LL_ENDL;
|
|
}
|
|
}
|
|
|
|
// static
|
|
void LLPanelLogin::onClickGrids()
|
|
{
|
|
//LLFloaterPreference::overrideLastTab(LLPreferenceCore::TAB_GRIDS);
|
|
LLFloaterPreference::show(NULL);
|
|
LLFloaterPreference::switchTab(LLPreferenceCore::TAB_GRIDS);
|
|
}
|
|
|
|
// static
|
|
void LLPanelLogin::onClickForgotPassword()
|
|
{
|
|
const std::string& url = gHippoGridManager->getCurrentGrid()->getPasswordUrl();
|
|
if (!url.empty())
|
|
LLWeb::loadURLExternal(url);
|
|
else
|
|
LL_WARNS() << "Link for 'forgotton password' not set." << LL_ENDL;
|
|
}
|
|
|
|
// static
|
|
void LLPanelLogin::onPassKey()
|
|
{
|
|
static bool sCapslockDidNotification = false;
|
|
if (gKeyboard->getKeyDown(KEY_CAPSLOCK) && sCapslockDidNotification == false)
|
|
{
|
|
LLNotificationsUtil::add("CapsKeyOn");
|
|
sCapslockDidNotification = true;
|
|
}
|
|
}
|
|
|
|
void LLPanelLogin::onCurGridChange(HippoGridInfo* new_grid, HippoGridInfo* old_grid)
|
|
{
|
|
refreshLoginPage();
|
|
if (old_grid != new_grid) //Changed grid? Reset the location combobox
|
|
{
|
|
std::string defaultStartLocation = gSavedSettings.getString("LoginLocation");
|
|
LLSLURL defaultStart(defaultStartLocation);
|
|
LLStartUp::setStartSLURL(defaultStart.isSpatial() ? defaultStart : LLSLURL(LLSLURL::SIM_LOCATION_HOME)); // calls onUpdateStartSLURL
|
|
}
|
|
}
|
|
|
|
// static
|
|
//void LLPanelLogin::updateServer()
|
|
void LLPanelLogin::refreshLoginPage()
|
|
{
|
|
if (!sInstance || (LLStartUp::getStartupState() >= STATE_LOGIN_CLEANUP))
|
|
return;
|
|
|
|
sInstance->updateGridCombo();
|
|
|
|
sInstance->getChildView("create_new_account_text")->setVisible(!gHippoGridManager->getCurrentGrid()->getRegisterUrl().empty());
|
|
sInstance->getChildView("forgot_password_text")->setVisible(!gHippoGridManager->getCurrentGrid()->getPasswordUrl().empty());
|
|
|
|
std::string login_page = gHippoGridManager->getCurrentGrid()->getLoginPage();
|
|
if (!login_page.empty())
|
|
{
|
|
LLMediaCtrl* web_browser = sInstance->getChild<LLMediaCtrl>("login_html");
|
|
if (web_browser->getCurrentNavUrl() != login_page)
|
|
{
|
|
//Singu note: No idea if site is alive, but we no longer check before loading.
|
|
sInstance->setSiteIsAlive(true);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
sInstance->setSiteIsAlive(false);
|
|
}
|
|
}
|
|
|
|
// static
|
|
//void LLPanelLogin::onSelectServer()
|
|
void LLPanelLogin::onSelectGrid(LLUICtrl *ctrl)
|
|
{
|
|
std::string grid(ctrl->getValue().asString());
|
|
LLStringUtil::trim(grid); // Guard against copy paste
|
|
if (!gHippoGridManager->getGrid(grid)) // We can't get an input grid by name or nick, perhaps a Login URI was entered
|
|
{
|
|
HippoGridInfo* info(new HippoGridInfo(LLStringUtil::null)); // Start off with empty grid name, otherwise we don't know what to name
|
|
info->setLoginUri(grid);
|
|
try
|
|
{
|
|
info->getGridInfo();
|
|
|
|
grid = info->getGridName();
|
|
if (HippoGridInfo* nick_info = gHippoGridManager->getGrid(info->getGridNick())) // Grid of same nick exists
|
|
{
|
|
delete info;
|
|
grid = nick_info->getGridName();
|
|
}
|
|
else // Guess not, try adding this grid
|
|
{
|
|
gHippoGridManager->addGrid(info); // deletes info if not needed (existing or no name)
|
|
}
|
|
}
|
|
catch(AIAlert::ErrorCode const& error)
|
|
{
|
|
// Inform the user of the problem, but only if something was entered that at least looks like a Login URI.
|
|
std::string::size_type pos1 = grid.find('.');
|
|
std::string::size_type pos2 = grid.find_last_of(".:");
|
|
if (grid.substr(0, 4) == "http" || (pos1 != std::string::npos && pos1 != pos2))
|
|
{
|
|
if (error.getCode() == HTTP_METHOD_NOT_ALLOWED || error.getCode() == HTTP_OK)
|
|
{
|
|
AIAlert::add("GridInfoError", error);
|
|
}
|
|
else
|
|
{
|
|
// Append GridInfoErrorInstruction to error message.
|
|
AIAlert::add("GridInfoError", AIAlert::Error(AIAlert::Prefix(), AIAlert::not_modal, error, "GridInfoErrorInstruction"));
|
|
}
|
|
}
|
|
delete info;
|
|
grid = gHippoGridManager->getCurrentGridName();
|
|
}
|
|
}
|
|
gHippoGridManager->setCurrentGrid(grid);
|
|
ctrl->setValue(grid);
|
|
sInstance->addFavoritesToStartLocation();
|
|
|
|
/*
|
|
* Determine whether or not the value in the start_location_combo makes sense
|
|
* with the new grid value.
|
|
*
|
|
* Note that some forms that could be in the location combo are grid-agnostic,
|
|
* such as "MyRegion/128/128/0". There could be regions with that name on any
|
|
* number of grids, so leave them alone. Other forms, such as
|
|
* https://grid.example.com/region/Party%20Town/20/30/5 specify a particular
|
|
* grid; in those cases we want to clear the location.
|
|
*/
|
|
auto location_combo = sInstance->getChild<LLComboBox>("start_location_combo");
|
|
S32 index = location_combo->getCurrentIndex();
|
|
switch (index)
|
|
{
|
|
case 0: // last location
|
|
case 1: // home location
|
|
// do nothing - these are grid-agnostic locations
|
|
break;
|
|
|
|
default:
|
|
{
|
|
std::string location = location_combo->getValue().asString();
|
|
LLSLURL slurl(location); // generata a slurl from the location combo contents
|
|
if ( slurl.getType() == LLSLURL::LOCATION
|
|
&& slurl.getGrid() != gHippoGridManager->getCurrentGridNick()
|
|
)
|
|
{
|
|
// the grid specified by the location is not this one, so clear the combo
|
|
location_combo->setCurrentByIndex(0); // last location on the new grid
|
|
location_combo->setTextEntry(LLStringUtil::null);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
void LLPanelLogin::onLocationSLURL()
|
|
{
|
|
auto location_combo = getChild<LLComboBox>("start_location_combo");
|
|
std::string location = location_combo->getValue().asString();
|
|
LLStringUtil::trim(location);
|
|
LL_DEBUGS("AppInit")<<location<<LL_ENDL;
|
|
|
|
LLStartUp::setStartSLURL(location); // calls onUpdateStartSLURL, above
|
|
}
|
|
|
|
//Special handling of name combobox. Facilitates grid-changing by account selection.
|
|
void LLPanelLogin::onSelectLoginEntry(const LLSD& selected_entry)
|
|
{
|
|
if (selected_entry.isMap())
|
|
setFields(LLSavedLoginEntry(selected_entry));
|
|
// This stops the automatic matching of the first name to a selected grid.
|
|
LLViewerLogin::getInstance()->setNameEditted(true);
|
|
|
|
sInstance->addFavoritesToStartLocation();
|
|
}
|
|
|
|
void LLPanelLogin::onLoginComboLostFocus(LLComboBox* combo_box)
|
|
{
|
|
if (combo_box->isTextDirty())
|
|
{
|
|
clearPassword();
|
|
combo_box->resetTextDirty();
|
|
}
|
|
}
|
|
|
|
void LLPanelLogin::onNameCheckChanged(const LLSD& value)
|
|
{
|
|
if (LLCheckBoxCtrl* remember_pass_check = findChild<LLCheckBoxCtrl>("remember_check"))
|
|
{
|
|
if (value.asBoolean())
|
|
{
|
|
remember_pass_check->setEnabled(true);
|
|
}
|
|
else
|
|
{
|
|
remember_pass_check->setValue(LLSD(false));
|
|
remember_pass_check->setEnabled(false);
|
|
}
|
|
}
|
|
}
|
|
|
|
void LLPanelLogin::confirmDelete()
|
|
{
|
|
LLNotificationsUtil::add("ConfirmDeleteUser", LLSD(), LLSD(), boost::bind(&LLPanelLogin::removeLogin, this, boost::bind(LLNotificationsUtil::getSelectedOption, _1, _2)));
|
|
}
|
|
|
|
void LLPanelLogin::removeLogin(bool knot)
|
|
{
|
|
if (knot) return;
|
|
LLComboBox* combo(getChild<LLComboBox>("username_combo"));
|
|
const std::string label(combo->getTextEntry());
|
|
if (combo->isTextDirty() || !combo->itemExists(label)) return; // Text entries aren't in the list
|
|
const LLSD& selected = combo->getSelectedValue();
|
|
if (!selected.isUndefined())
|
|
{
|
|
mLoginHistoryData.deleteEntry(selected.get("firstname").asString(), selected.get("lastname").asString(), selected.get("grid").asString());
|
|
combo->remove(label);
|
|
combo->selectFirstItem();
|
|
}
|
|
}
|