llpanellogin.cpp
上传用户:king477883
上传日期:2021-03-01
资源大小:9553k
文件大小:27k
源码类别:

游戏引擎

开发平台:

C++ Builder

  1. /** 
  2.  * @file llpanellogin.cpp
  3.  * @brief Login dialog and logo display
  4.  *
  5.  * $LicenseInfo:firstyear=2002&license=viewergpl$
  6.  * 
  7.  * Copyright (c) 2002-2010, Linden Research, Inc.
  8.  * 
  9.  * Second Life Viewer Source Code
  10.  * The source code in this file ("Source Code") is provided by Linden Lab
  11.  * to you under the terms of the GNU General Public License, version 2.0
  12.  * ("GPL"), unless you have obtained a separate licensing agreement
  13.  * ("Other License"), formally executed by you and Linden Lab.  Terms of
  14.  * the GPL can be found in doc/GPL-license.txt in this distribution, or
  15.  * online at http://secondlifegrid.net/programs/open_source/licensing/gplv2
  16.  * 
  17.  * There are special exceptions to the terms and conditions of the GPL as
  18.  * it is applied to this Source Code. View the full text of the exception
  19.  * in the file doc/FLOSS-exception.txt in this software distribution, or
  20.  * online at
  21.  * http://secondlifegrid.net/programs/open_source/licensing/flossexception
  22.  * 
  23.  * By copying, modifying or distributing this software, you acknowledge
  24.  * that you have read and understood your obligations described above,
  25.  * and agree to abide by those obligations.
  26.  * 
  27.  * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO
  28.  * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY,
  29.  * COMPLETENESS OR PERFORMANCE.
  30.  * $/LicenseInfo$
  31.  */
  32. #include "llviewerprecompiledheaders.h"
  33. #include "llpanellogin.h"
  34. #include "indra_constants.h" // for key and mask constants
  35. #include "llfloaterreg.h"
  36. #include "llfontgl.h"
  37. #include "llmd5.h"
  38. #include "llsecondlifeurls.h"
  39. #include "v4color.h"
  40. #include "llbutton.h"
  41. #include "llcheckboxctrl.h"
  42. #include "llcommandhandler.h" // for secondlife:///app/login/
  43. #include "llcombobox.h"
  44. #include "llcurl.h"
  45. #include "llviewercontrol.h"
  46. #include "llfloaterpreference.h"
  47. #include "llfocusmgr.h"
  48. #include "lllineeditor.h"
  49. #include "llnotificationsutil.h"
  50. #include "llstartup.h"
  51. #include "lltextbox.h"
  52. #include "llui.h"
  53. #include "lluiconstants.h"
  54. #include "llurlsimstring.h"
  55. #include "llversioninfo.h"
  56. #include "llviewerhelp.h"
  57. #include "llviewertexturelist.h"
  58. #include "llviewermenu.h" // for handle_preferences()
  59. #include "llviewernetwork.h"
  60. #include "llviewerwindow.h" // to link into child list
  61. #include "lluictrlfactory.h"
  62. #include "llhttpclient.h"
  63. #include "llweb.h"
  64. #include "llmediactrl.h"
  65. #include "llrootview.h"
  66. #include "llfloatertos.h"
  67. #include "lltrans.h"
  68. #include "llglheaders.h"
  69. #include "llpanelloginlistener.h"
  70. #if LL_WINDOWS
  71. #pragma warning(disable: 4355)      // 'this' used in initializer list
  72. #endif  // LL_WINDOWS
  73. #define USE_VIEWER_AUTH 0
  74. const S32 BLACK_BORDER_HEIGHT = 160;
  75. const S32 MAX_PASSWORD = 16;
  76. LLPanelLogin *LLPanelLogin::sInstance = NULL;
  77. BOOL LLPanelLogin::sCapslockDidNotification = FALSE;
  78. class LLLoginRefreshHandler : public LLCommandHandler
  79. {
  80. public:
  81. // don't allow from external browsers
  82. LLLoginRefreshHandler() : LLCommandHandler("login_refresh", UNTRUSTED_BLOCK) { }
  83. bool handle(const LLSD& tokens, const LLSD& query_map, LLMediaCtrl* web)
  84. {
  85. if (LLStartUp::getStartupState() < STATE_LOGIN_CLEANUP)
  86. {
  87. LLPanelLogin::loadLoginPage();
  88. }
  89. return true;
  90. }
  91. };
  92. LLLoginRefreshHandler gLoginRefreshHandler;
  93. // helper class that trys to download a URL from a web site and calls a method 
  94. // on parent class indicating if the web server is working or not
  95. class LLIamHereLogin : public LLHTTPClient::Responder
  96. {
  97. private:
  98. LLIamHereLogin( LLPanelLogin* parent ) :
  99.    mParent( parent )
  100. {}
  101. LLPanelLogin* mParent;
  102. public:
  103. static boost::intrusive_ptr< LLIamHereLogin > build( LLPanelLogin* parent )
  104. {
  105. return boost::intrusive_ptr< LLIamHereLogin >( new LLIamHereLogin( parent ) );
  106. };
  107. virtual void  setParent( LLPanelLogin* parentIn )
  108. {
  109. mParent = parentIn;
  110. };
  111. // We don't actually expect LLSD back, so need to override completedRaw
  112. virtual void completedRaw(U32 status, const std::string& reason,
  113.   const LLChannelDescriptors& channels,
  114.   const LLIOPipe::buffer_ptr_t& buffer)
  115. {
  116. completed(status, reason, LLSD()); // will call result() or error()
  117. }
  118. virtual void result( const LLSD& content )
  119. {
  120. if ( mParent )
  121. mParent->setSiteIsAlive( true );
  122. };
  123. virtual void error( U32 status, const std::string& reason )
  124. {
  125. if ( mParent )
  126. mParent->setSiteIsAlive( false );
  127. };
  128. };
  129. // this is global and not a class member to keep crud out of the header file
  130. namespace {
  131. boost::intrusive_ptr< LLIamHereLogin > gResponsePtr = 0;
  132. };
  133. void set_start_location(LLUICtrl* ctrl, void* data)
  134. {
  135.     LLURLSimString::setString(ctrl->getValue().asString());
  136. }
  137. //---------------------------------------------------------------------------
  138. // Public methods
  139. //---------------------------------------------------------------------------
  140. LLPanelLogin::LLPanelLogin(const LLRect &rect,
  141.  BOOL show_server,
  142.  void (*callback)(S32 option, void* user_data),
  143.  void *cb_data)
  144. : LLPanel(),
  145. mLogoImage(),
  146. mCallback(callback),
  147. mCallbackData(cb_data),
  148. mHtmlAvailable( TRUE ),
  149. mListener(new LLPanelLoginListener(this))
  150. {
  151. setFocusRoot(TRUE);
  152. setBackgroundVisible(FALSE);
  153. setBackgroundOpaque(TRUE);
  154. // instance management
  155. if (LLPanelLogin::sInstance)
  156. {
  157. llwarns << "Duplicate instance of login view deleted" << llendl;
  158. // Don't leave bad pointer in gFocusMgr
  159. gFocusMgr.setDefaultKeyboardFocus(NULL);
  160. delete LLPanelLogin::sInstance;
  161. }
  162. LLPanelLogin::sInstance = this;
  163. // add to front so we are the bottom-most child
  164. gViewerWindow->getRootView()->addChildInBack(this);
  165. // Logo
  166. mLogoImage = LLUI::getUIImage("startup_logo");
  167. LLUICtrlFactory::getInstance()->buildPanel(this, "panel_login.xml");
  168. #if USE_VIEWER_AUTH
  169. //leave room for the login menu bar
  170. setRect(LLRect(0, rect.getHeight()-18, rect.getWidth(), 0)); 
  171. #endif
  172. // Legacy login web page is hidden under the menu bar.
  173. // Adjust reg-in-client web browser widget to not be hidden.
  174. if (gSavedSettings.getBOOL("RegInClient"))
  175. {
  176. reshape(rect.getWidth(), rect.getHeight() - MENU_BAR_HEIGHT);
  177. }
  178. else
  179. {
  180. reshape(rect.getWidth(), rect.getHeight());
  181. }
  182. #if !USE_VIEWER_AUTH
  183. childSetPrevalidate("first_name_edit", LLTextValidate::validateASCIIPrintableNoSpace);
  184. childSetPrevalidate("last_name_edit", LLTextValidate::validateASCIIPrintableNoSpace);
  185. childSetCommitCallback("password_edit", mungePassword, this);
  186. getChild<LLLineEditor>("password_edit")->setKeystrokeCallback(onPassKey, this);
  187. // change z sort of clickable text to be behind buttons
  188. sendChildToBack(getChildView("channel_text"));
  189. sendChildToBack(getChildView("forgot_password_text"));
  190. LLLineEditor* edit = getChild<LLLineEditor>("password_edit");
  191. if (edit) edit->setDrawAsterixes(TRUE);
  192. LLComboBox* combo = getChild<LLComboBox>("start_location_combo");
  193. std::string sim_string = LLURLSimString::sInstance.mSimString;
  194. if(sim_string.empty())
  195. {
  196. LLURLSimString::setString(gSavedSettings.getString("LoginLocation"));
  197. }
  198. if (!sim_string.empty())
  199. {
  200. // Replace "<Type region name>" with this region name
  201. combo->remove(2);
  202. combo->add( sim_string );
  203. combo->setTextEntry(sim_string);
  204. combo->setCurrentByIndex( 2 );
  205. }
  206. combo->setCommitCallback( &set_start_location, NULL );
  207. LLComboBox* server_choice_combo = sInstance->getChild<LLComboBox>("server_combo");
  208. server_choice_combo->setCommitCallback(onSelectServer, NULL);
  209. server_choice_combo->setFocusLostCallback(boost::bind(onServerComboLostFocus, _1));
  210. childSetAction("connect_btn", onClickConnect, this);
  211. getChild<LLPanel>("login")->setDefaultBtn("connect_btn");
  212. std::string channel = gSavedSettings.getString("VersionChannelName");
  213. std::string version = llformat("%s (%d)",
  214.    LLVersionInfo::getShortVersion().c_str(),
  215.    LLVersionInfo::getBuild());
  216. LLTextBox* channel_text = getChild<LLTextBox>("channel_text");
  217. channel_text->setTextArg("[CHANNEL]", channel); // though not displayed
  218. channel_text->setTextArg("[VERSION]", version);
  219. channel_text->setClickedCallback(onClickVersion, this);
  220. LLTextBox* forgot_password_text = getChild<LLTextBox>("forgot_password_text");
  221. forgot_password_text->setClickedCallback(onClickForgotPassword, NULL);
  222. LLTextBox* create_new_account_text = getChild<LLTextBox>("create_new_account_text");
  223. create_new_account_text->setClickedCallback(onClickNewAccount, NULL);
  224. LLTextBox* need_help_text = getChild<LLTextBox>("login_help");
  225. need_help_text->setClickedCallback(onClickHelp, NULL);
  226. #endif    
  227. // get the web browser control
  228. LLMediaCtrl* web_browser = getChild<LLMediaCtrl>("login_html");
  229. web_browser->addObserver(this);
  230. // Clear the browser's cache to avoid any potential for the cache messing up the login screen.
  231. web_browser->clearCache();
  232. // Need to handle login secondlife:///app/ URLs
  233. web_browser->setTrusted( true );
  234. // don't make it a tab stop until SL-27594 is fixed
  235. web_browser->setTabStop(FALSE);
  236. // web_browser->navigateToLocalPage( "loading", "loading.html" );
  237. if (gSavedSettings.getBOOL("RegInClient"))
  238. {
  239. // need to follow links in the internal browser
  240. web_browser->setOpenInExternalBrowser( false );
  241. getChild<LLView>("login_widgets")->setVisible(false);
  242. }
  243. else
  244. {
  245. // make links open in external browser
  246. web_browser->setOpenInExternalBrowser( true );
  247. reshapeBrowser();
  248. }
  249. // kick off a request to grab the url manually
  250. gResponsePtr = LLIamHereLogin::build( this );
  251. std::string login_page = gSavedSettings.getString("LoginPage");
  252. if (login_page.empty())
  253. {
  254. login_page = getString( "real_url" );
  255. }
  256. LLHTTPClient::head( login_page, gResponsePtr );
  257. #if !USE_VIEWER_AUTH
  258. // Initialize visibility (and don't force visibility - use prefs)
  259. refreshLocation( false );
  260. #endif
  261. }
  262. // force the size to be correct (XML doesn't seem to be sufficient to do this)
  263. // (with some padding so the other login screen doesn't show through)
  264. void LLPanelLogin::reshapeBrowser()
  265. {
  266. LLMediaCtrl* web_browser = getChild<LLMediaCtrl>("login_html");
  267. LLRect rect = gViewerWindow->getWindowRectScaled();
  268. LLRect html_rect;
  269. #if USE_VIEWER_AUTH
  270. html_rect.setCenterAndSize( 
  271. rect.getCenterX() - 2, rect.getCenterY(), 
  272. rect.getWidth() + 6, rect.getHeight());
  273. #else
  274. html_rect.setCenterAndSize(
  275. rect.getCenterX() - 2, rect.getCenterY() + 40,
  276. rect.getWidth() + 6, rect.getHeight() - 78 );
  277. #endif
  278. web_browser->setRect( html_rect );
  279. web_browser->reshape( html_rect.getWidth(), html_rect.getHeight(), TRUE );
  280. reshape( rect.getWidth(), rect.getHeight(), 1 );
  281. }
  282. void LLPanelLogin::setSiteIsAlive( bool alive )
  283. {
  284. LLMediaCtrl* web_browser = getChild<LLMediaCtrl>("login_html");
  285. // if the contents of the site was retrieved
  286. if ( alive )
  287. {
  288. if ( web_browser )
  289. {
  290. loadLoginPage();
  291. // mark as available
  292. mHtmlAvailable = TRUE;
  293. }
  294. }
  295. else
  296. // the site is not available (missing page, server down, other badness)
  297. {
  298. #if !USE_VIEWER_AUTH
  299. if ( web_browser )
  300. {
  301. // hide browser control (revealing default one)
  302. web_browser->setVisible( FALSE );
  303. // mark as unavailable
  304. mHtmlAvailable = FALSE;
  305. }
  306. #else
  307. if ( web_browser )
  308. {
  309. web_browser->navigateToLocalPage( "loading-error" , "index.html" );
  310. // mark as available
  311. mHtmlAvailable = TRUE;
  312. }
  313. #endif
  314. }
  315. }
  316. void LLPanelLogin::mungePassword(LLUICtrl* caller, void* user_data)
  317. {
  318. LLPanelLogin* self = (LLPanelLogin*)user_data;
  319. LLLineEditor* editor = (LLLineEditor*)caller;
  320. std::string password = editor->getText();
  321. // Re-md5 if we've changed at all
  322. if (password != self->mIncomingPassword)
  323. {
  324. LLMD5 pass((unsigned char *)password.c_str());
  325. char munged_password[MD5HEX_STR_SIZE];
  326. pass.hex_digest(munged_password);
  327. self->mMungedPassword = munged_password;
  328. }
  329. }
  330. LLPanelLogin::~LLPanelLogin()
  331. {
  332. LLPanelLogin::sInstance = NULL;
  333. // tell the responder we're not here anymore
  334. if ( gResponsePtr )
  335. gResponsePtr->setParent( 0 );
  336. //// We know we're done with the image, so be rid of it.
  337. //gTextureList.deleteImage( mLogoImage );
  338. // Controls having keyboard focus by default
  339. // must reset it on destroy. (EXT-2748)
  340. gFocusMgr.setDefaultKeyboardFocus(NULL);
  341. }
  342. // virtual
  343. void LLPanelLogin::draw()
  344. {
  345. glPushMatrix();
  346. {
  347. F32 image_aspect = 1.333333f;
  348. F32 view_aspect = (F32)getRect().getWidth() / (F32)getRect().getHeight();
  349. // stretch image to maintain aspect ratio
  350. if (image_aspect > view_aspect)
  351. {
  352. glTranslatef(-0.5f * (image_aspect / view_aspect - 1.f) * getRect().getWidth(), 0.f, 0.f);
  353. glScalef(image_aspect / view_aspect, 1.f, 1.f);
  354. }
  355. S32 width = getRect().getWidth();
  356. S32 height = getRect().getHeight();
  357. if ( mHtmlAvailable )
  358. {
  359. #if !USE_VIEWER_AUTH
  360. if (getChild<LLView>("login_widgets")->getVisible())
  361. {
  362. // draw a background box in black
  363. gl_rect_2d( 0, height - 264, width, 264, LLColor4::black );
  364. // draw the bottom part of the background image
  365. // just the blue background to the native client UI
  366. mLogoImage->draw(0, -264, width + 8, mLogoImage->getHeight());
  367. }
  368. #endif
  369. }
  370. else
  371. {
  372. // the HTML login page is not available so default to the original screen
  373. S32 offscreen_part = height / 3;
  374. mLogoImage->draw(0, -offscreen_part, width, height+offscreen_part);
  375. };
  376. }
  377. glPopMatrix();
  378. LLPanel::draw();
  379. }
  380. // virtual
  381. BOOL LLPanelLogin::handleKeyHere(KEY key, MASK mask)
  382. {
  383. if (( KEY_RETURN == key ) && (MASK_ALT == mask))
  384. {
  385. gViewerWindow->toggleFullscreen(FALSE);
  386. return TRUE;
  387. }
  388. if ( KEY_F1 == key )
  389. {
  390. LLViewerHelp* vhelp = LLViewerHelp::getInstance();
  391. vhelp->showTopic(vhelp->f1HelpTopic());
  392. return TRUE;
  393. }
  394. return LLPanel::handleKeyHere(key, mask);
  395. }
  396. // virtual 
  397. void LLPanelLogin::setFocus(BOOL b)
  398. {
  399. if(b != hasFocus())
  400. {
  401. if(b)
  402. {
  403. LLPanelLogin::giveFocus();
  404. }
  405. else
  406. {
  407. LLPanel::setFocus(b);
  408. }
  409. }
  410. }
  411. // static
  412. void LLPanelLogin::giveFocus()
  413. {
  414. #if USE_VIEWER_AUTH
  415. if (sInstance)
  416. {
  417. sInstance->setFocus(TRUE);
  418. }
  419. #else
  420. if( sInstance )
  421. {
  422. // Grab focus and move cursor to first blank input field
  423. std::string first = sInstance->childGetText("first_name_edit");
  424. std::string pass = sInstance->childGetText("password_edit");
  425. BOOL have_first = !first.empty();
  426. BOOL have_pass = !pass.empty();
  427. LLLineEditor* edit = NULL;
  428. if (have_first && !have_pass)
  429. {
  430. // User saved his name but not his password.  Move
  431. // focus to password field.
  432. edit = sInstance->getChild<LLLineEditor>("password_edit");
  433. }
  434. else
  435. {
  436. // User doesn't have a name, so start there.
  437. edit = sInstance->getChild<LLLineEditor>("first_name_edit");
  438. }
  439. if (edit)
  440. {
  441. edit->setFocus(TRUE);
  442. edit->selectAll();
  443. }
  444. }
  445. #endif
  446. }
  447. // static
  448. void LLPanelLogin::showLoginWidgets()
  449. {
  450. sInstance->childSetVisible("login_widgets", true);
  451. LLMediaCtrl* web_browser = sInstance->getChild<LLMediaCtrl>("login_html");
  452. web_browser->setOpenInExternalBrowser( true );
  453. sInstance->reshapeBrowser();
  454. // *TODO: Append all the usual login parameters, like first_login=Y etc.
  455. std::string splash_screen_url = sInstance->getString("real_url");
  456. web_browser->navigateTo( splash_screen_url, "text/html" );
  457. LLUICtrl* first_name_edit = sInstance->getChild<LLUICtrl>("first_name_edit");
  458. first_name_edit->setFocus(TRUE);
  459. }
  460. // static
  461. void LLPanelLogin::show(const LLRect &rect,
  462. BOOL show_server,
  463. void (*callback)(S32 option, void* user_data),
  464. void* callback_data)
  465. {
  466. new LLPanelLogin(rect, show_server, callback, callback_data);
  467. if( !gFocusMgr.getKeyboardFocus() )
  468. {
  469. // Grab focus and move cursor to first enabled control
  470. sInstance->setFocus(TRUE);
  471. }
  472. // Make sure that focus always goes here (and use the latest sInstance that was just created)
  473. gFocusMgr.setDefaultKeyboardFocus(sInstance);
  474. }
  475. // static
  476. void LLPanelLogin::setFields(const std::string& firstname,
  477.      const std::string& lastname,
  478.      const std::string& password)
  479. {
  480. if (!sInstance)
  481. {
  482. llwarns << "Attempted fillFields with no login view shown" << llendl;
  483. return;
  484. }
  485. sInstance->childSetText("first_name_edit", firstname);
  486. sInstance->childSetText("last_name_edit", lastname);
  487. // Max "actual" password length is 16 characters.
  488. // Hex digests are always 32 characters.
  489. if (password.length() == 32)
  490. {
  491. // This is a MD5 hex digest of a password.
  492. // We don't actually use the password input field, 
  493. // fill it with MAX_PASSWORD characters so we get a 
  494. // nice row of asterixes.
  495. const std::string filler("123456789!123456");
  496. sInstance->childSetText("password_edit", filler);
  497. sInstance->mIncomingPassword = filler;
  498. sInstance->mMungedPassword = password;
  499. }
  500. else
  501. {
  502. // this is a normal text password
  503. sInstance->childSetText("password_edit", password);
  504. sInstance->mIncomingPassword = password;
  505. LLMD5 pass((unsigned char *)password.c_str());
  506. char munged_password[MD5HEX_STR_SIZE];
  507. pass.hex_digest(munged_password);
  508. sInstance->mMungedPassword = munged_password;
  509. }
  510. }
  511. // static
  512. void LLPanelLogin::addServer(const std::string& server, S32 domain_name)
  513. {
  514. if (!sInstance)
  515. {
  516. llwarns << "Attempted addServer with no login view shown" << llendl;
  517. return;
  518. }
  519. LLComboBox* combo = sInstance->getChild<LLComboBox>("server_combo");
  520. combo->add(server, LLSD(domain_name) );
  521. combo->setCurrentByIndex(0);
  522. }
  523. // static
  524. void LLPanelLogin::getFields(std::string *firstname,
  525.      std::string *lastname,
  526.      std::string *password)
  527. {
  528. if (!sInstance)
  529. {
  530. llwarns << "Attempted getFields with no login view shown" << llendl;
  531. return;
  532. }
  533. *firstname = sInstance->childGetText("first_name_edit");
  534. LLStringUtil::trim(*firstname);
  535. *lastname = sInstance->childGetText("last_name_edit");
  536. LLStringUtil::trim(*lastname);
  537. *password = sInstance->mMungedPassword;
  538. }
  539. // static
  540. BOOL LLPanelLogin::isGridComboDirty()
  541. {
  542. BOOL user_picked = FALSE;
  543. if (!sInstance)
  544. {
  545. llwarns << "Attempted getServer with no login view shown" << llendl;
  546. }
  547. else
  548. {
  549. LLComboBox* combo = sInstance->getChild<LLComboBox>("server_combo");
  550. user_picked = combo->isDirty();
  551. }
  552. return user_picked;
  553. }
  554. // static
  555. void LLPanelLogin::getLocation(std::string &location)
  556. {
  557. if (!sInstance)
  558. {
  559. llwarns << "Attempted getLocation with no login view shown" << llendl;
  560. return;
  561. }
  562. LLComboBox* combo = sInstance->getChild<LLComboBox>("start_location_combo");
  563. location = combo->getValue().asString();
  564. }
  565. // static
  566. void LLPanelLogin::refreshLocation( bool force_visible )
  567. {
  568. if (!sInstance) return;
  569. #if USE_VIEWER_AUTH
  570. loadLoginPage();
  571. #else
  572. BOOL show_start = TRUE;
  573. if ( ! force_visible )
  574. {
  575. // Don't show on first run after install
  576. // Otherwise ShowStartLocation defaults to true.
  577. show_start = gSavedSettings.getBOOL("ShowStartLocation");
  578. }
  579. sInstance->childSetVisible("start_location_combo", show_start);
  580. sInstance->childSetVisible("start_location_text", show_start);
  581. BOOL show_server = gSavedSettings.getBOOL("ForceShowGrid");
  582. sInstance->childSetVisible("server_combo", show_server);
  583. #endif
  584. }
  585. // static
  586. void LLPanelLogin::updateLocationUI()
  587. {
  588. if (!sInstance) return;
  589. std::string sim_string = LLURLSimString::sInstance.mSimString;
  590. if (!sim_string.empty())
  591. {
  592. // Replace "<Type region name>" with this region name
  593. LLComboBox* combo = sInstance->getChild<LLComboBox>("start_location_combo");
  594. combo->remove(2);
  595. combo->add( sim_string );
  596. combo->setTextEntry(sim_string);
  597. combo->setCurrentByIndex( 2 );
  598. }
  599. }
  600. // static
  601. void LLPanelLogin::closePanel()
  602. {
  603. if (sInstance)
  604. {
  605. gViewerWindow->getRootView()->removeChild( LLPanelLogin::sInstance );
  606. delete sInstance;
  607. sInstance = NULL;
  608. }
  609. }
  610. // static
  611. void LLPanelLogin::setAlwaysRefresh(bool refresh)
  612. {
  613. if (LLStartUp::getStartupState() >= STATE_LOGIN_CLEANUP) return;
  614. LLMediaCtrl* web_browser = sInstance->getChild<LLMediaCtrl>("login_html");
  615. if (web_browser)
  616. {
  617. web_browser->setAlwaysRefresh(refresh);
  618. }
  619. }
  620. void LLPanelLogin::loadLoginPage()
  621. {
  622. if (!sInstance) return;
  623. std::ostringstream oStr;
  624. std::string login_page = gSavedSettings.getString("LoginPage");
  625. if (login_page.empty())
  626. {
  627. login_page = sInstance->getString( "real_url" );
  628. }
  629. oStr << login_page;
  630. // Use the right delimeter depending on how LLURI parses the URL
  631. LLURI login_page_uri = LLURI(login_page);
  632. std::string first_query_delimiter = "&";
  633. if (login_page_uri.queryMap().size() == 0)
  634. {
  635. first_query_delimiter = "?";
  636. }
  637. // Language
  638. std::string language = LLUI::getLanguage();
  639. oStr << first_query_delimiter<<"lang=" << language;
  640. // First Login?
  641. if (gSavedSettings.getBOOL("FirstLoginThisInstall"))
  642. {
  643. oStr << "&firstlogin=TRUE";
  644. }
  645. // Channel and Version
  646. std::string version = llformat("%s (%d)",
  647.    LLVersionInfo::getShortVersion().c_str(),
  648.    LLVersionInfo::getBuild());
  649. char* curl_channel = curl_escape(gSavedSettings.getString("VersionChannelName").c_str(), 0);
  650. char* curl_version = curl_escape(version.c_str(), 0);
  651. oStr << "&channel=" << curl_channel;
  652. oStr << "&version=" << curl_version;
  653. curl_free(curl_channel);
  654. curl_free(curl_version);
  655. // Grid
  656. char* curl_grid = curl_escape(LLViewerLogin::getInstance()->getGridLabel().c_str(), 0);
  657. oStr << "&grid=" << curl_grid;
  658. curl_free(curl_grid);
  659. gViewerWindow->setMenuBackgroundColor(false, !LLViewerLogin::getInstance()->isInProductionGrid());
  660. gLoginMenuBarView->setBackgroundColor(gMenuBarView->getBackgroundColor());
  661. #if USE_VIEWER_AUTH
  662. LLURLSimString::sInstance.parse();
  663. std::string location;
  664. std::string region;
  665. std::string password;
  666. if (LLURLSimString::parse())
  667. {
  668. std::ostringstream oRegionStr;
  669. location = "specify";
  670. oRegionStr << LLURLSimString::sInstance.mSimName << "/" << LLURLSimString::sInstance.mX << "/"
  671.  << LLURLSimString::sInstance.mY << "/"
  672.  << LLURLSimString::sInstance.mZ;
  673. region = oRegionStr.str();
  674. }
  675. else
  676. {
  677. location = gSavedSettings.getString("LoginLocation");
  678. }
  679. std::string firstname, lastname;
  680.     if(gSavedSettings.getLLSD("UserLoginInfo").size() == 3)
  681.     {
  682.         LLSD cmd_line_login = gSavedSettings.getLLSD("UserLoginInfo");
  683. firstname = cmd_line_login[0].asString();
  684. lastname = cmd_line_login[1].asString();
  685.         password = cmd_line_login[2].asString();
  686.     }
  687.     
  688. if (firstname.empty())
  689. {
  690. firstname = gSavedSettings.getString("FirstName");
  691. }
  692. if (lastname.empty())
  693. {
  694. lastname = gSavedSettings.getString("LastName");
  695. }
  696. char* curl_region = curl_escape(region.c_str(), 0);
  697. oStr <<"firstname=" << firstname <<
  698. "&lastname=" << lastname << "&location=" << location << "&region=" << curl_region;
  699. curl_free(curl_region);
  700. if (!password.empty())
  701. {
  702. oStr << "&password=" << password;
  703. }
  704. else if (!(password = load_password_from_disk()).empty())
  705. {
  706. oStr << "&password=$1$" << password;
  707. }
  708. if (gAutoLogin)
  709. {
  710. oStr << "&auto_login=TRUE";
  711. }
  712. if (gSavedSettings.getBOOL("ShowStartLocation"))
  713. {
  714. oStr << "&show_start_location=TRUE";
  715. }
  716. if (gSavedSettings.getBOOL("RememberPassword"))
  717. {
  718. oStr << "&remember_password=TRUE";
  719. }
  720. #ifndef LL_RELEASE_FOR_DOWNLOAD
  721. oStr << "&show_grid=TRUE";
  722. #else
  723. if (gSavedSettings.getBOOL("ForceShowGrid"))
  724. oStr << "&show_grid=TRUE";
  725. #endif
  726. #endif
  727. LLMediaCtrl* web_browser = sInstance->getChild<LLMediaCtrl>("login_html");
  728. // navigate to the "real" page
  729. if (gSavedSettings.getBOOL("RegInClient"))
  730. {
  731. web_browser->setFocus(TRUE);
  732. login_page = sInstance->getString("reg_in_client_url");
  733. web_browser->navigateTo(login_page, "text/html");
  734. }
  735. else
  736. {
  737. web_browser->navigateTo( oStr.str(), "text/html" );
  738. }
  739. }
  740. void LLPanelLogin::handleMediaEvent(LLPluginClassMedia* /*self*/, EMediaEvent event)
  741. {
  742. if(event == MEDIA_EVENT_NAVIGATE_COMPLETE)
  743. {
  744. LLMediaCtrl* web_browser = sInstance->getChild<LLMediaCtrl>("login_html");
  745. if (web_browser)
  746. {
  747. // *HACK HACK HACK HACK!
  748. /* Stuff a Tab key into the browser now so that the first field will
  749. ** get the focus!  The embedded javascript on the page that properly
  750. ** sets the initial focus in a real web browser is not working inside
  751. ** the viewer, so this is an UGLY HACK WORKAROUND for now.
  752. */
  753. // Commented out as it's not reliable
  754. //web_browser->handleKey(KEY_TAB, MASK_NONE, false);
  755. }
  756. }
  757. }
  758. //---------------------------------------------------------------------------
  759. // Protected methods
  760. //---------------------------------------------------------------------------
  761. // static
  762. void LLPanelLogin::onClickConnect(void *)
  763. {
  764. if (sInstance && sInstance->mCallback)
  765. {
  766. // tell the responder we're not here anymore
  767. if ( gResponsePtr )
  768. gResponsePtr->setParent( 0 );
  769. // JC - Make sure the fields all get committed.
  770. sInstance->setFocus(FALSE);
  771. std::string first = sInstance->childGetText("first_name_edit");
  772. std::string last  = sInstance->childGetText("last_name_edit");
  773. LLComboBox* combo = sInstance->getChild<LLComboBox>("start_location_combo");
  774. std::string combo_text = combo->getSimple();
  775. bool has_first_and_last = !(first.empty() || last.empty());
  776. bool has_location = false;
  777. if(combo_text=="<Type region name>" || combo_text =="")
  778. {
  779. // *NOTE: Mani - Location field is not always committed by this point!
  780. // This may be duplicate work, but better than not doing the work!
  781. LLURLSimString::sInstance.setString("");
  782. }
  783. else 
  784. {
  785. // *NOTE: Mani - Location field is not always committed by this point!
  786. LLURLSimString::sInstance.setString(combo_text);
  787. has_location = true;
  788. }
  789. if(!has_first_and_last)
  790. {
  791. LLNotificationsUtil::add("MustHaveAccountToLogIn");
  792. }
  793. else if(!has_location)
  794. {
  795. LLNotificationsUtil::add("StartRegionEmpty");
  796. }
  797. else
  798. {
  799. // has both first and last name typed
  800. sInstance->mCallback(0, sInstance->mCallbackData);
  801. }
  802. }
  803. }
  804. /*
  805. // static
  806. bool LLPanelLogin::newAccountAlertCallback(const LLSD& notification, const LLSD& response)
  807. {
  808. S32 option = LLNotificationsUtil::getSelectedOption(notification, response);
  809. if (0 == option)
  810. {
  811. llinfos << "Going to account creation URL" << llendl;
  812. LLWeb::loadURLExternal( LLNotifications::instance().getGlobalString("CREATE_ACCOUNT_URL")); 
  813. }
  814. else
  815. {
  816. sInstance->setFocus(TRUE);
  817. }
  818. return false;
  819. }
  820. */
  821. // static
  822. void LLPanelLogin::onClickNewAccount(void*)
  823. {
  824. LLWeb::loadURLExternal(sInstance->getString("create_account_url"));
  825. }
  826. // static
  827. void LLPanelLogin::onClickVersion(void*)
  828. {
  829. LLFloaterReg::showInstance("sl_about"); 
  830. }
  831. //static
  832. void LLPanelLogin::onClickForgotPassword(void*)
  833. {
  834. if (sInstance )
  835. {
  836. LLWeb::loadURLExternal(sInstance->getString( "forgot_password_url" ));
  837. }
  838. }
  839. //static
  840. void LLPanelLogin::onClickHelp(void*)
  841. {
  842. if (sInstance)
  843. {
  844. LLViewerHelp* vhelp = LLViewerHelp::getInstance();
  845. vhelp->showTopic(vhelp->preLoginTopic());
  846. }
  847. }
  848. // static
  849. void LLPanelLogin::onPassKey(LLLineEditor* caller, void* user_data)
  850. {
  851. if (gKeyboard->getKeyDown(KEY_CAPSLOCK) && sCapslockDidNotification == FALSE)
  852. {
  853. LLNotificationsUtil::add("CapsKeyOn");
  854. sCapslockDidNotification = TRUE;
  855. }
  856. }
  857. // static
  858. void LLPanelLogin::onSelectServer(LLUICtrl*, void*)
  859. {
  860. // *NOTE: The paramters for this method are ignored. 
  861. // LLPanelLogin::onServerComboLostFocus(LLFocusableElement* fe, void*)
  862. // calls this method.
  863. // The user twiddled with the grid choice ui.
  864. // apply the selection to the grid setting.
  865. std::string grid_label;
  866. S32 grid_index;
  867. LLComboBox* combo = sInstance->getChild<LLComboBox>("server_combo");
  868. LLSD combo_val = combo->getValue();
  869. if (LLSD::TypeInteger == combo_val.type())
  870. {
  871. grid_index = combo->getValue().asInteger();
  872. if ((S32)GRID_INFO_OTHER == grid_index)
  873. {
  874. // This happens if the user specifies a custom grid
  875. // via command line.
  876. grid_label = combo->getSimple();
  877. }
  878. }
  879. else
  880. {
  881. // no valid selection, return other
  882. grid_index = (S32)GRID_INFO_OTHER;
  883. grid_label = combo_val.asString();
  884. }
  885. // This new seelction will override preset uris
  886. // from the command line.
  887. LLViewerLogin* vl = LLViewerLogin::getInstance();
  888. vl->resetURIs();
  889. if(grid_index != GRID_INFO_OTHER)
  890. {
  891. vl->setGridChoice((EGridInfo)grid_index);
  892. }
  893. else
  894. {
  895. vl->setGridChoice(grid_label);
  896. }
  897. // grid changed so show new splash screen (possibly)
  898. loadLoginPage();
  899. }
  900. void LLPanelLogin::onServerComboLostFocus(LLFocusableElement* fe)
  901. {
  902. if (!sInstance) return;
  903. LLComboBox* combo = sInstance->getChild<LLComboBox>("server_combo");
  904. if(fe == combo)
  905. {
  906. onSelectServer(combo, NULL);
  907. }
  908. }