2020-09-30 17:12:32 +02:00

1321 lines
43 KiB
C++

/***
MultiUI.cpp
Code for handling multiple user interface in IE
and friends
Initially by Christopher Evans (cevans) 4/28/98
****/
#include "private.h"
#include "resource.h"
#include "multiui.h"
#include "multiutl.h"
#include "multiusr.h"
#include "mluisup.h"
#include "strconst.h"
#include "commctrl.h"
extern HINSTANCE g_hInst;
static const GUID GUID_NULL = { /* 00000000-0000-0000-0000-000000000000 */
0x0,
0x0,
0x0,
{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0}
};
static const HELPMAP g_rgCtxMapMultiUserGeneral[] = {
{IDC_NO_HELP_1, NO_HELP},
{IDC_NO_HELP_2, NO_HELP},
{IDC_NO_HELP_3, NO_HELP},
{IDC_NO_HELP_4, NO_HELP},
{idcWarningIcon, NO_HELP},
{idcConfirmMsg, NO_HELP},
{idcErrorMsg, NO_HELP},
{idcLoginInstr, NO_HELP},
{idcWelcomeMsg, NO_HELP},
{idcAdd, IDH_IDENTITY_ADD},
{idcNewPwd, IDH_IDENTITY_PWORD_NEW},
{idcPwdCaption,IDH_IDENTITY_ENTER_PWORD},
{idcPwd, IDH_IDENTITY_ENTER_PWORD},
{idcProperties, IDH_IDENTITY_PROPERTIES},
{idcConfPwd, IDH_IDENTITY_CONFIRM_PWORD},
{idcUserName, IDH_IDENTITY_NAME},
{idcDefault, IDH_IDENTITY_DEFAULT},
{idcOldPwd, IDH_IDENTITY_PWORD_OLD},
{idcStartupCombo, IDH_IDENTITY_STARTAS},
{idcDelete, IDH_IDENTITY_DELETE},
{idcStaticName, IDH_IDENTITY_LIST},
{idcUserNameList, IDH_IDENTITY_LIST},
{idcTellMeMore, /*IDH_IDENTITY_TELLMEMORE_CONTENT */IDH_IDENTITY_TELLMEMORE},
{idcStaticNames, IDH_IDENTITY_LIST},
{idcStaticStartUp, IDH_IDENTITY_STARTAS},
{idcUsePwd, IDH_IDENTITY_PROMPT_PWORD},
{idcChgPwd, IDH_IDENTITY_CHANGE_PWORD},
{idcConfirmPwd, IDH_MULTI_DELETE_PWORD},
{idcManage, IDH_IDENTITY_MANAGE},
{idcLogoff, IDH_MULTI_LOG_OFF},
{idcCheckDefault, IDH_MULTI_MNG_IDENT_DEFAULT},
{idcDefaultCombo, IDH_MULTI_MNG_DEFAULT_LIST},
{0,0}};
/*
MU_ShowErrorMessage
Simple wrapper around resource string table based call to MessageBox
*/
void MU_ShowErrorMessage(HWND hwnd, UINT iMsgID, UINT iTitleID)
{
TCHAR szMsg[255], szTitle[63];
MLLoadStringA(iMsgID, szMsg, ARRAYSIZE(szMsg));
MLLoadStringA(iTitleID, szTitle, ARRAYSIZE(szTitle));
MessageBox(hwnd, szMsg, szTitle, MB_OK | MB_ICONEXCLAMATION);
}
/*
_StripDefault
Remove the (Default) string from the user's name, if it
appears. Should be called after getting a username from
the listbox since the default user has the string (Default)
appended to it
*/
void _StripDefault(LPSTR psz)
{
TCHAR szResString[CCH_USERNAME_MAX_LENGTH], *pszStr;
MLLoadStringA(idsDefault, szResString, CCH_USERNAME_MAX_LENGTH);
pszStr = strstr(psz, szResString);
if(pszStr)
{
*pszStr = 0;
}
}
#ifdef IDENTITY_PASSWORDS
// **
// C H A N G E U S E R P A S S W O R D
/*
_ValidateChangePasswordValues
Validate the data entered by the user. Return true only if everything is
legitimate,
*/
static BOOL _ValidateChangePasswordValues(HWND hDlg,
TCHAR* lpszOldNewPassword)
{
TCHAR szOldPW[255], szPW1[255], szPW2[255];
GetDlgItemText(hDlg,idcOldPwd, szOldPW, ARRAYSIZE(szOldPW));
GetDlgItemText(hDlg,idcNewPwd, szPW1, ARRAYSIZE(szPW1));
GetDlgItemText(hDlg,idcConfPwd, szPW2, ARRAYSIZE(szPW2));
if (strcmp(lpszOldNewPassword, szOldPW) != 0)
{
MU_ShowErrorMessage(hDlg, idsPwdDoesntMatch, idsPwdError);
SetFocus(GetDlgItem(hDlg,idcOldPwd));
SendDlgItemMessage(hDlg,idcOldPwd,EM_SETSEL,0,-1);
return false;
}
if (strcmp(szPW1, szPW2) != 0)
{
MU_ShowErrorMessage(hDlg, idsPwdChgNotMatch, idsPwdError);
SetFocus(GetDlgItem(hDlg,idcNewPwd));
SendDlgItemMessage(hDlg,idcNewPwd,EM_SETSEL,0,-1);
return false;
}
strcpy(lpszOldNewPassword, szPW1);
return true;
}
/*
_ChangeUserPwdDlgProc
Description: Dialog proc for handling the change user password dialog.
*/
INT_PTR CALLBACK _ChangeUserPwdDlgProc(HWND hDlg,
UINT iMsg,
WPARAM wParam,
LPARAM lParam)
{
static TCHAR *sOldNewPassword;
switch (iMsg)
{
case WM_INITDIALOG:
SendMessage(GetDlgItem(hDlg, idcNewPwd), EM_LIMITTEXT, CCH_USERPASSWORD_MAX_LENGTH-1, 0);
SendMessage(GetDlgItem(hDlg, idcOldPwd), EM_LIMITTEXT, CCH_USERPASSWORD_MAX_LENGTH-1, 0);
SendMessage(GetDlgItem(hDlg,idcConfPwd), EM_LIMITTEXT, CCH_USERPASSWORD_MAX_LENGTH-1, 0);
sOldNewPassword = (TCHAR *)lParam;
return TRUE;
case WM_HELP:
case WM_CONTEXTMENU:
return OnContextHelp(hDlg, iMsg, wParam, lParam, g_rgCtxMapMultiUserGeneral);
case WM_COMMAND:
switch(LOWORD(wParam))
{
case IDOK:
if (_ValidateChangePasswordValues(hDlg, sOldNewPassword))
MLEndDialogWrap(hDlg, IDOK);
return true;
case IDCANCEL:
MLEndDialogWrap(hDlg, IDCANCEL);
return true;
}
break;
}
return false;
}
/*
ChangeUserPassword
Wrapper routine for changing the user password. Pass in the current
password in lpszOldNewPassword which is used to confirm the current
password with what the user entered. If the user enters the old
password correctly and enters the new password twice correctly,
and clicks OK, then the new password is returned in lpszOldNewPassword
and this function returns TRUE. Otherwise, the value in lpszOldNewPassword
is unchanged and it returns false.
lpszOldNewPassword must point to a TCHAR buffer large enough to hold a password
(CCH_USERPASSWORD_MAX_LENGTH characters)
*/
BOOL ChangeUserPassword(HWND hwnd, TCHAR *lpszOldNewPassword)
{
INT_PTR bResult;
Assert(hwnd);
Assert(lpszOldNewPassword);
bResult = MLDialogBoxParamWrap(MLGetHinst(), MAKEINTRESOURCEW(iddChgPwd), hwnd, _ChangeUserPwdDlgProc, (LPARAM)lpszOldNewPassword);
//Don't actually change it here, the caller will do the right thing
//since this may be (and is) called from another dialog with a cancel
//button on it
return (bResult == IDOK);
}
// **
// C O N F I R M U S E R P A S S W O R D
/*
_ConfirmUserPwdDlgProc
Description: Dialog proc for handling the confirming user password dialog.
*/
INT_PTR CALLBACK _ConfirmUserPwdDlgProc(HWND hDlg,
UINT iMsg,
WPARAM wParam,
LPARAM lParam)
{
static LPCONFIRMPWDDIALOGINFO sConfirmPwdInfo;
switch (iMsg)
{
case WM_INITDIALOG:
Assert(lParam);
SendMessage(GetDlgItem(hDlg, idcConfirmPwd), EM_LIMITTEXT, CCH_USERPASSWORD_MAX_LENGTH-1, 0);
sConfirmPwdInfo = (LPCONFIRMPWDDIALOGINFO)lParam;
SetDlgItemText(hDlg, idcConfirmMsg, sConfirmPwdInfo->szMsg);
return TRUE;
case WM_HELP:
case WM_CONTEXTMENU:
return OnContextHelp(hDlg, iMsg, wParam, lParam, g_rgCtxMapMultiUserGeneral);
case WM_COMMAND:
switch(LOWORD(wParam))
{
case IDOK:
TCHAR szPW[255];
//if the password matches the provided password, then
//everything is OK and the dialog can complete, otherwise,
//barf an error message and keep waiting for a good password
//or Cancel.
GetDlgItemText(hDlg,idcConfirmPwd, szPW, ARRAYSIZE(szPW));
if (strcmp(szPW, sConfirmPwdInfo->szPassword) == 0)
MLEndDialogWrap(hDlg, IDOK);
else
{
MU_ShowErrorMessage(hDlg, idsPwdDoesntMatch, idsPwdError);
SetFocus(GetDlgItem(hDlg,idcConfirmPwd));
SendDlgItemMessage(hDlg,idcConfirmPwd,EM_SETSEL,0,-1);
}
return true;
case IDCANCEL:
MLEndDialogWrap(hDlg, IDCANCEL);
return true;
}
break;
}
return false;
}
/*
MU_ConfirmUserPassword
Confirm that the user knows the password before it is disabled
in the registry. If they enter the correct password, simply return
true since the calling dialog box will do the right thing if the
user clicks cancel there.
*/
BOOL MU_ConfirmUserPassword(HWND hwnd, TCHAR *lpszMsg, TCHAR *lpszPassword)
{
INT_PTR bResult;
CONFIRMPWDDIALOGINFO vConfirmInfo;
Assert(hwnd);
Assert(lpszPassword);
Assert(lpszMsg);
Assert(lstrlen(lpszMsg) < ARRAYSIZE(vConfirmInfo.szMsg));
Assert(lstrlen(lpszPassword) < ARRAYSIZE(vConfirmInfo.szPassword));
strcpy(vConfirmInfo.szMsg, lpszMsg);
strcpy(vConfirmInfo.szPassword, lpszPassword);
bResult = MLDialogBoxParamWrap(MLGetHinst(), MAKEINTRESOURCEW(iddPasswordOff), hwnd, _ConfirmUserPwdDlgProc, (LPARAM)&vConfirmInfo);
return (bResult == IDOK);
}
// **
// E N T E R U S E R P A S S W O R D
/*
_ValidateNewPasswordValues
Description: Make sure that the entered values in the new password
dialog are legit and consistant.
*/
static BOOL _ValidateNewPasswordValues(HWND hDlg,
TCHAR* lpszNewPassword)
{
TCHAR szPW1[255], szPW2[255];
GetDlgItemText(hDlg,idcNewPwd, szPW1, ARRAYSIZE(szPW1));
GetDlgItemText(hDlg,idcConfPwd, szPW2, ARRAYSIZE(szPW2));
if (strcmp(szPW1, szPW2) != 0)
{
MU_ShowErrorMessage(hDlg, idsPwdChgNotMatch, idsPwdError);
SetFocus(GetDlgItem(hDlg,idcNewPwd));
SendDlgItemMessage(hDlg,idcNewPwd,EM_SETSEL,0,-1);
return false;
}
strcpy(lpszNewPassword, szPW1);
return true;
}
/*
_EnterUserPwdDlgProc
Description: Dialog proc for handling the enter user password dialog.
*/
INT_PTR CALLBACK _EnterUserPwdDlgProc(HWND hDlg,
UINT iMsg,
WPARAM wParam,
LPARAM lParam)
{
static TCHAR *sNewPassword;
switch (iMsg)
{
case WM_INITDIALOG:
SendMessage(GetDlgItem(hDlg, idcNewPwd), EM_LIMITTEXT, CCH_USERPASSWORD_MAX_LENGTH-1, 0);
SendMessage(GetDlgItem(hDlg, idcConfPwd), EM_LIMITTEXT, CCH_USERPASSWORD_MAX_LENGTH-1, 0);
sNewPassword = (TCHAR *)lParam;
return TRUE;
case WM_HELP:
case WM_CONTEXTMENU:
return OnContextHelp(hDlg, iMsg, wParam, lParam, g_rgCtxMapMultiUserGeneral);
case WM_COMMAND:
switch(LOWORD(wParam))
{
case IDOK:
if (_ValidateNewPasswordValues(hDlg, sNewPassword))
MLEndDialogWrap(hDlg, IDOK);
return true;
case IDCANCEL:
MLEndDialogWrap(hDlg, IDCANCEL);
return true;
}
break;
}
return false;
}
/*
EnterUserPassword
Wrapper routine for getting a new user password. If the user enters the
password and confirms it correctly, and clicks OK, then the new password is
returned in lpszNewPassword and this function returns TRUE.
Otherwise, the value in lpszNewPassword is unchanged and it returns false.
lpszNewPassword must point to a TCHAR buffer large enough to hold a password
(CCH_USERPASSWORD_MAX_LENGTH characters)
*/
BOOL EnterUserPassword(HWND hwnd, TCHAR *lpszNewPassword)
{
INT_PTR bResult;
Assert(hwnd);
Assert(lpszNewPassword);
bResult = MLDialogBoxParamWrap(MLGetHinst(), MAKEINTRESOURCEW(iddNewPwd), hwnd, _EnterUserPwdDlgProc, (LPARAM)lpszNewPassword);
return (bResult == IDOK);
}
#endif //IDENTITY_PASSWORDS
// **
// C O N F I R M D E L E T E U S E R D I A L O G
/*
ConfirmDeleteUserDlgProc
Description: Dialog proc for handling the confirm delete user dialog.
*/
INT_PTR CALLBACK _ConfirmDeleteUserDlgProc(HWND hDlg,
UINT iMsg,
WPARAM wParam,
LPARAM lParam)
{
switch (iMsg)
{
case WM_INITDIALOG:
Assert(lParam);
SendDlgItemMessage(hDlg, idcWarningIcon, STM_SETICON, (WPARAM)::LoadIcon(NULL, IDI_EXCLAMATION), 0);
SetDlgItemText(hDlg, idcErrorMsg, (TCHAR *)lParam);
return TRUE;
case WM_HELP:
case WM_CONTEXTMENU:
return OnContextHelp(hDlg, iMsg, wParam, lParam, g_rgCtxMapMultiUserGeneral);
case WM_COMMAND:
switch(LOWORD(wParam))
{
case IDOK:
case IDCANCEL:
MLEndDialogWrap(hDlg, LOWORD(wParam));
return true;
}
break;
}
return false;
}
BOOL MU_ConfirmDeleteUser(HWND hwnd, TCHAR *lpszUsername)
{
LPTSTR lpString = NULL;
TCHAR szUsername[63];
TCHAR* rgdw[1] = {szUsername};
TCHAR szBuffer[255]; // really ought to be big enough
TCHAR szPassword[CCH_USERPASSWORD_MAX_LENGTH];
strcpy(szUsername, lpszUsername);
// format the message with the username scattered throughout.
MLLoadStringA(idsConfirmDeleteMsg, szBuffer, ARRAYSIZE(szBuffer));
if (szBuffer[0])
{
FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER |
FORMAT_MESSAGE_FROM_STRING |
FORMAT_MESSAGE_ARGUMENT_ARRAY,
szBuffer,
0, 0,
(LPTSTR)&lpString, 0, (va_list *)rgdw);
}
if (lpString)
{
INT_PTR bResult;
// Show the Confirm Delete dialog box to make sure they really want to delete the user
bResult = MLDialogBoxParamWrap(MLGetHinst(), MAKEINTRESOURCEW(iddConfirmUserDelete), hwnd, _ConfirmDeleteUserDlgProc, (LPARAM)lpString);
LocalFree(lpString);
#ifdef IDENTITY_PASSWORDS
if (IDOK == bResult)
{
BOOL fUsePassword;
// check to see if this user has a password, if so, then make sure that
// they know the password before blowing it all away.
if (MU_GetPasswordForUsername(lpszUsername, szPassword, &fUsePassword))
{
if (fUsePassword)
{
MLLoadStringA(idsConfirmDelPwd, szBuffer, ARRAYSIZE(szBuffer));
FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER |
FORMAT_MESSAGE_FROM_STRING |
FORMAT_MESSAGE_ARGUMENT_ARRAY,
szBuffer,
0, 0,
(LPTSTR)&lpString, 0, (va_list *)rgdw);
if (lpString)
{
if (!MU_ConfirmUserPassword(hwnd,lpString, szPassword))
bResult = IDCANCEL;
LocalFree(lpString);
}
else
{
bResult = IDCANCEL;
}
}
}
else //couldn't load the password, can't delete them either
{
MU_ShowErrorMessage(hwnd, idsPwdNotFound, idsPwdError);
bResult = IDCANCEL;
}
return (IDOK == bResult);
}
#else
return (IDOK == bResult);
#endif //IDENTITY_PASSWORDS
}
return false;
}
// **
// C H A N G E U S E R S E T T I N G S
/*
_ValidateChangeUserValues
Validate the data entered by the user. Return true only if everything is
legit,
*/
static BOOL _ValidateChangeUserValues(HWND hDlg,
LPUSERINFO lpUserInfo)
{
TCHAR szResString[CCH_USERNAME_MAX_LENGTH], *pszStr;
TCHAR szUsername[255];
ULONG cb;
GetDlgItemText(hDlg,idcUserName, szUsername, ARRAYSIZE(szUsername));
cb = lstrlen(szUsername);
UlStripWhitespace(szUsername, false, true, &cb); //remove trailing whitespace
// Make sure the username wasn't all spaces
if (!cb)
{
MU_ShowErrorMessage(hDlg, idsUserNameTooShort, idsNameTooShort);
SetFocus(GetDlgItem(hDlg,idcUserName));
SendDlgItemMessage(hDlg,idcUserName,EM_SETSEL,0,-1);
return false;
}
// if the username exists, and its not the same as the account currently, then
// it is not allowed.
if (MU_UsernameExists(szUsername) && strcmp(szUsername, lpUserInfo->szUsername) != 0)
{
MU_ShowErrorMessage(hDlg, idsUserNameExists, idsUserNameInUse);
SetFocus(GetDlgItem(hDlg,idcUserName));
SendDlgItemMessage(hDlg,idcUserName,EM_SETSEL,0,-1);
return false;
}
lstrcpy(lpUserInfo->szUsername, szUsername);
lpUserInfo->fUsePassword = IsDlgButtonChecked(hDlg, idcUsePwd);
if (!lpUserInfo->fUsePassword)
lpUserInfo->szPassword[0] = 0;
return true;
}
/*
ChangeUserSettingsDlgProc
Description: Dialog proc for handling the Change user settings dialog.
*/
INT_PTR CALLBACK _ChangeUserSettingsDlgProc(HWND hDlg,
UINT iMsg,
WPARAM wParam,
LPARAM lParam)
{
static LPUSERINFO sUserInfo;
TCHAR szMsg[255];
TCHAR szPassword[CCH_USERPASSWORD_MAX_LENGTH];
switch (iMsg)
{
case WM_INITDIALOG:
Assert(lParam);
sUserInfo = (LPUSERINFO)lParam;
MLLoadStringA((*sUserInfo->szUsername) ? idsIdentityProperties : idsNewIdentity, szMsg, ARRAYSIZE(szMsg));
SendMessage(hDlg, WM_SETTEXT, 0, (LPARAM)szMsg);
SetDlgItemText(hDlg, idcUserName, sUserInfo->szUsername);
SendMessage(GetDlgItem(hDlg, idcUserName), EM_LIMITTEXT, CCH_IDENTITY_NAME_MAX_LENGTH/2, 0);
CheckDlgButton(hDlg, idcUsePwd, sUserInfo->fUsePassword ? BST_CHECKED : BST_UNCHECKED);
EnableWindow(GetDlgItem(hDlg, idcChgPwd), sUserInfo->fUsePassword);
// Don't allow zero length names by disabling OK
if (!lstrlen(sUserInfo->szUsername))
EnableWindow(GetDlgItem(hDlg, IDOK), FALSE);
return TRUE;
case WM_HELP:
case WM_CONTEXTMENU:
return OnContextHelp(hDlg, iMsg, wParam, lParam, g_rgCtxMapMultiUserGeneral);
case WM_COMMAND:
switch(LOWORD(wParam))
{
case IDOK:
if (_ValidateChangeUserValues(hDlg, sUserInfo))
MLEndDialogWrap(hDlg, IDOK);
return true;
case IDCANCEL:
MLEndDialogWrap(hDlg, IDCANCEL);
return true;
case idcUserName:
if (EN_CHANGE == HIWORD(wParam))
{
EnableWindow(GetDlgItem(hDlg, IDOK), SendMessage((HWND)lParam, WM_GETTEXTLENGTH, 0, 0) != 0);
return TRUE;
}
break;
#ifdef IDENTITY_PASSWORDS
case idcTellMeMore:
// WinHelp ((HWND)GetDlgItem(hDlg, idcTellMeMore),
// c_szCtxHelpFile,
// HELP_WM_HELP,
// (DWORD_PTR)(LPVOID)g_rgCtxMapMultiUserGeneral);
WinHelp(hDlg, c_szCtxHelpFile, HELP_CONTEXT, IDH_IDENTITY_TELLMEMORE_CONTENT);
return true;
case idcUsePwd:
// if they are turning off the password, they need to confirm it first.
if (!IsDlgButtonChecked(hDlg, idcUsePwd))
{
strcpy(szPassword, sUserInfo->szPassword);
MLLoadStringA(idsConfirmDisablePwd, szMsg, ARRAYSIZE(szMsg));
if (!MU_ConfirmUserPassword(hDlg,szMsg, szPassword))
CheckDlgButton(hDlg, idcUsePwd, BST_CHECKED);
}
else
{
// if they are turning it on, they should set the password.
if (EnterUserPassword(hDlg, szPassword))
{
sUserInfo->fUsePassword = true;
strcpy(sUserInfo->szPassword, szPassword);
}
else
{
CheckDlgButton(hDlg, idcUsePwd, BST_UNCHECKED);
}
}
EnableWindow(GetDlgItem(hDlg, idcChgPwd), IsDlgButtonChecked(hDlg, idcUsePwd));
return true;
case idcChgPwd:
if(sUserInfo->fUsePassword || (0 != *sUserInfo->szPassword))
{
strcpy(szPassword, sUserInfo->szPassword);
if (ChangeUserPassword(hDlg, szPassword))
strcpy(sUserInfo->szPassword, szPassword);
}
return true;
#endif //IDENTITY_PASSWORDS
}
break;
}
return false;
}
/*
MU_UserProperties
Allow the user the change their username or password.
*/
BOOL MU_UserProperties(HWND hwnd, LPUSERINFO lpUserInfo)
{
INT_PTR fResult;
USERINFO nuInfo;
TCHAR szOldUsername[CCH_IDENTITY_NAME_MAX_LENGTH+1];
USERINFO uiCurrent;
LPARAM lpNotify = IIC_CURRENT_IDENTITY_CHANGED;
INITCOMMONCONTROLSEX icex;
Assert(hwnd);
Assert(lpUserInfo);
// get the current info so we know who to change later.
MU_GetUserInfo(NULL, &nuInfo);
lstrcpy(szOldUsername, lpUserInfo->szUsername);
// make sure ICC_NATIVEFNTCTL_CLASS is inited
icex.dwSize = sizeof(INITCOMMONCONTROLSEX);
icex.dwICC = ICC_NATIVEFNTCTL_CLASS;
InitCommonControlsEx(&icex);
fResult = MLDialogBoxParamWrap(MLGetHinst(), MAKEINTRESOURCEW(iddUserProperties), hwnd, _ChangeUserSettingsDlgProc, (LPARAM)lpUserInfo);
if (IDOK == fResult)
{
if (GUID_NULL == lpUserInfo->uidUserID)
_ClaimNextUserId(&lpUserInfo->uidUserID);
MU_SetUserInfo(lpUserInfo);
// if its not the current identity, then just broadcast that an identity changed
if (MU_GetUserInfo(NULL, &uiCurrent) && (lpUserInfo->uidUserID != uiCurrent.uidUserID))
lpNotify = IIC_IDENTITY_CHANGED;
// if the name changd, tell other apps
if (lstrcmp(szOldUsername, lpUserInfo->szUsername) != 0)
PostMessage(HWND_BROADCAST, WM_IDENTITY_INFO_CHANGED, 0, lpNotify);
}
return (IDOK == fResult);
}
// **
// L O G I N S C R E E N
/*
_ValidateLoginValues
Validate the data entered by the user. Return true only if everything is
legit,
*/
static BOOL _ValidateLoginValues(HWND hDlg,
TCHAR* lpszOldNewPassword)
{
TCHAR szUsername[255];
TCHAR szPW[255], szRealPW[CCH_USERPASSWORD_MAX_LENGTH];
LRESULT dSelItem;
BOOL rResult = false;
dSelItem = SendDlgItemMessage(hDlg, idcUserNameList, LB_GETCURSEL, 0, 0);
if (LB_ERR != dSelItem)
{
if (SendDlgItemMessage(hDlg, idcUserNameList, LB_GETTEXTLEN, dSelItem, 0) < ARRAYSIZE(szUsername))
{
SendDlgItemMessage(hDlg, idcUserNameList, LB_GETTEXT, dSelItem, (LPARAM)szUsername);
#ifdef IDENTITY_PASSWORDS
BOOL fUsePassword;
if (MU_GetPasswordForUsername(szUsername, szRealPW, &fUsePassword))
{
if (fUsePassword)
{
GetDlgItemText(hDlg,idcPwd,szPW, ARRAYSIZE(szPW));
if (strcmp(szPW, szRealPW) == 0)
{
strcpy(lpszOldNewPassword, szUsername);
rResult = true;
}
else
{
MU_ShowErrorMessage(hDlg, idsPwdDoesntMatch, idsPwdError);
SetFocus(GetDlgItem(hDlg,idcPwd));
SendDlgItemMessage(hDlg,idcPwd,EM_SETSEL,0,-1);
return false;
}
}
else // if there is no password, then it does match up.
{
strcpy(lpszOldNewPassword, szUsername);
rResult = true;
}
}
else //can't load identity password, do not allow access
{
MU_ShowErrorMessage(hDlg, idsPwdNotFound, idsPwdError);
return false;
}
#else //IDENTITY_PASSWORDS
strcpy(lpszOldNewPassword, szUsername);
rResult = true;
#endif //IDENTITY_PASSWORDS
}
}
return rResult;
}
static void _LoginEnableDisablePwdField(HWND hDlg)
{
#ifdef IDENTITY_PASSWORDS
TCHAR szUsername[255], szRealPW[255];
BOOL bEnabled = false;
#endif //IDENTITY_PASSWORDS
LRESULT dSelItem;
dSelItem = SendDlgItemMessage(hDlg, idcUserNameList, LB_GETCURSEL, 0, 0);
#ifdef IDENTITY_PASSWORDS
if (LB_ERR != dSelItem)
{
if (SendDlgItemMessage(hDlg, idcUserNameList, LB_GETTEXTLEN, dSelItem, 0) < ARRAYSIZE(szUsername))
{
SendDlgItemMessage(hDlg, idcUserNameList, LB_GETTEXT, dSelItem, (LPARAM)szUsername);
BOOL fUsePassword;
if (MU_GetPasswordForUsername(szUsername, szRealPW, &fUsePassword) && fUsePassword)
{
bEnabled = true;
}
}
}
EnableWindow(GetDlgItem(hDlg,idcPwd),bEnabled);
EnableWindow(GetDlgItem(hDlg,idcPwdCaption),bEnabled);
#endif //IDENTITY_PASSWORDS
EnableWindow(GetDlgItem(hDlg,IDOK),(dSelItem != -1));
}
typedef struct
{
TCHAR *pszUsername;
DWORD dwFlags;
} LOGIN_PARAMS;
/*
_LoginDlgProc
Description: Dialog proc for handling the OE Login dialog.
*/
INT_PTR CALLBACK _LoginDlgProc(HWND hDlg,
UINT iMsg,
WPARAM wParam,
LPARAM lParam)
{
static TCHAR *sResultUsername;
static LOGIN_PARAMS *plpParams;
TCHAR szMsg[1024], szRes[1024];
USERINFO nuInfo;
switch (iMsg)
{
case WM_INITDIALOG:
Assert(lParam);
plpParams = (LOGIN_PARAMS *)lParam;
sResultUsername = plpParams->pszUsername;
MLLoadStringA(!!(plpParams->dwFlags & UIL_FORCE_UI) ? idsSwitchIdentities : idsIdentityLogin, szMsg, ARRAYSIZE(szMsg));
SendMessage(hDlg, WM_SETTEXT, 0, (LPARAM)szMsg);
_FillListBoxWithUsernames(GetDlgItem(hDlg,idcUserNameList));
if (MU_GetUserInfo(NULL, &nuInfo))
{
MLLoadStringA(idsLoginWithCurrent, szRes, ARRAYSIZE(szRes));
wsprintf(szMsg, szRes, nuInfo.szUsername);
SetDlgItemText(hDlg, idcWelcomeMsg, szMsg);
MLLoadStringA(idsCurrIdentityInstr, szMsg, ARRAYSIZE(szMsg));
SetDlgItemText(hDlg, idcLoginInstr, szMsg);
}
else
{
MLLoadStringA(idsLoginNoCurrent, szMsg, ARRAYSIZE(szMsg));
SetDlgItemText(hDlg, idcWelcomeMsg, szMsg);
MLLoadStringA(idsNoIdentityInstr, szMsg, ARRAYSIZE(szMsg));
SetDlgItemText(hDlg, idcLoginInstr, szMsg);
}
if (sResultUsername[0] == 0)
strcpy(sResultUsername, nuInfo.szUsername);
if (sResultUsername[0])
{
LRESULT dFoundItem;
dFoundItem = SendDlgItemMessage(hDlg, idcUserNameList, LB_FINDSTRING, 0, (LPARAM)sResultUsername);
if (LB_ERR != dFoundItem)
{
SendDlgItemMessage(hDlg, idcUserNameList, LB_SETCURSEL, dFoundItem, 0);
}
}
else
SendDlgItemMessage(hDlg, idcUserNameList, LB_SETCURSEL, 0, 0);
_LoginEnableDisablePwdField(hDlg);
return TRUE;
case WM_HELP:
case WM_CONTEXTMENU:
return OnContextHelp(hDlg, iMsg, wParam, lParam, g_rgCtxMapMultiUserGeneral);
case WM_COMMAND:
switch(HIWORD(wParam))
{
case LBN_DBLCLK:
wParam = IDOK;
break;
case LBN_SELCHANGE:
_LoginEnableDisablePwdField(hDlg);
break;
}
switch(LOWORD(wParam))
{
case IDOK:
if (_ValidateLoginValues(hDlg, sResultUsername))
MLEndDialogWrap(hDlg, IDOK);
return true;
case IDCANCEL:
MLEndDialogWrap(hDlg, IDCANCEL);
return true;
case idcLogoff:
MLLoadStringA(idsLogoff, sResultUsername, CCH_USERNAME_MAX_LENGTH);
MLEndDialogWrap(hDlg, IDOK);
return true;
case idcManage:
{
TCHAR szUsername[CCH_USERNAME_MAX_LENGTH+1] = "";
MU_ManageUsers(hDlg, szUsername, 0);
_FillListBoxWithUsernames(GetDlgItem(hDlg,idcUserNameList));
SendDlgItemMessage(hDlg, idcUserNameList, LB_SETCURSEL, 0, 0);
_LoginEnableDisablePwdField(hDlg);
if (*szUsername)
{
lstrcpy(sResultUsername, szUsername);
MLEndDialogWrap(hDlg, IDOK);
}
}
return true;
}
break;
}
return false;
}
/*
MU_Login
Wrapper routine for logging in to OE. Asks the user to choose a username
and, if necessary, enter the password for that user. The user can also
create an account at this point.
lpszUsername should contain the name of the person who should be the default
selection in the list. If the name is empty ("") then it will look up the
default from the registry.
Returns the username that was selected in lpszUsername. Returns true
if that username is valid.
*/
BOOL MU_Login(HWND hwnd, DWORD dwFlags, TCHAR *lpszUsername)
{
INT_PTR bResult;
CStringList *csList;
INITCOMMONCONTROLSEX icex;
Assert(hwnd);
Assert(lpszUsername);
// make sure ICC_NATIVEFNTCTL_CLASS is inited
icex.dwSize = sizeof(INITCOMMONCONTROLSEX);
icex.dwICC = ICC_NATIVEFNTCTL_CLASS;
InitCommonControlsEx(&icex);
csList = MU_GetUsernameList();
// if there is only one username and they do not have a password, just return it.
if (csList && csList->GetLength() == 1 && !(dwFlags & UIL_FORCE_UI))
{
TCHAR *pszUsername;
TCHAR szPassword[255];
BOOL fUsePassword;
pszUsername = csList->GetString(0);
if(MU_GetPasswordForUsername(pszUsername, szPassword, &fUsePassword) && !fUsePassword)
{
lstrcpy(lpszUsername, pszUsername);
delete csList;
return TRUE;
}
}
LOGIN_PARAMS lpParams;
lpParams.dwFlags = dwFlags;
lpParams.pszUsername = lpszUsername;
bResult = MLDialogBoxParamWrap(MLGetHinst(), MAKEINTRESOURCEW(iddLogin), hwnd, _LoginDlgProc, (LPARAM)&lpParams);
if (csList)
delete csList;
return (IDOK == bResult);
}
void _ManagerUpdateButtons(HWND hDlg)
{
LRESULT dFoundItem;
USERINFO rUserInfo;
GUID uidDefaultId;
// make sure that the delete button is only available if the
// current user is not selected.
dFoundItem = SendDlgItemMessage(hDlg, idcUserNameList, LB_GETCURSEL, 0, 0);
if (dFoundItem != -1)
{
SendDlgItemMessage(hDlg, idcUserNameList, LB_GETTEXT, dFoundItem, (LPARAM)rUserInfo.szUsername);
MU_UsernameToUserId(rUserInfo.szUsername, &rUserInfo.uidUserID);
MU_GetCurrentUserID(&uidDefaultId);
// if there is no current user, don't allow deletion of the default user.
if (GUID_NULL == uidDefaultId)
MU_GetDefaultUserID(&uidDefaultId);
}
EnableWindow(GetDlgItem(hDlg, idcDelete), uidDefaultId != rUserInfo.uidUserID && dFoundItem != -1);
}
typedef struct
{
TCHAR *pszUsername;
DWORD dwFlags;
} MANAGE_PARAMS;
/*
_ManagerDlgProc
Description: Dialog proc for handling the identity manager dialog.
*/
INT_PTR CALLBACK _ManagerDlgProc(HWND hDlg,
UINT iMsg,
WPARAM wParam,
LPARAM lParam)
{
USERINFO rUserInfo;
static MANAGE_PARAMS *pmpParams;
static TCHAR sResultUsername[MAX_PATH] = "";
static DWORD sdwFlags = 0;
LRESULT dFoundItem;
ULONG uidUserId;
HRESULT hr;
TCHAR szRes[256];
USERINFO nuInfo;
DWORD dwIndex;
GUID uidDefault;
switch (iMsg)
{
case WM_INITDIALOG:
Assert(lParam);
_ResetRememberedLoginOption();
pmpParams = (MANAGE_PARAMS*)lParam;
sdwFlags = pmpParams->dwFlags;
_FillListBoxWithUsernames(GetDlgItem(hDlg,idcUserNameList));
_FillComboBoxWithUsernames(GetDlgItem(hDlg,idcStartupCombo), GetDlgItem(hDlg,idcUserNameList));
_FillComboBoxWithUsernames(GetDlgItem(hDlg,idcDefaultCombo), GetDlgItem(hDlg,idcUserNameList));
dwIndex = MU_GetLoginOptionIndex(GetDlgItem(hDlg,idcStartupCombo));
CheckDlgButton(hDlg, idcCheckDefault, dwIndex != ASK_BEFORE_LOGIN);
EnableWindow(GetDlgItem(hDlg, idcStartupCombo), dwIndex != ASK_BEFORE_LOGIN);
if (dwIndex != ASK_BEFORE_LOGIN)
SendDlgItemMessage(hDlg, idcStartupCombo, CB_SETCURSEL, dwIndex, 0);
else
SendDlgItemMessage(hDlg, idcStartupCombo, CB_SETCURSEL, 0, 0);
MU_GetUserInfo(NULL, &nuInfo);
strcpy(szRes, nuInfo.szUsername);
if (szRes[0])
{
dFoundItem = SendDlgItemMessage(hDlg, idcUserNameList, LB_FINDSTRING, 0, (LPARAM)szRes);
if (LB_ERR != dFoundItem)
{
SendDlgItemMessage(hDlg, idcUserNameList, LB_SETCURSEL, dFoundItem, 0);
}
}
SendDlgItemMessage(hDlg, idcDefaultCombo, CB_SETCURSEL, MU_GetDefaultOptionIndex(GetDlgItem(hDlg, idcDefaultCombo)), 0);
_ManagerUpdateButtons(hDlg);
if (!!(sdwFlags & UIMI_CREATE_NEW_IDENTITY))
{
ShowWindow(hDlg, SW_SHOW);
PostMessage(hDlg, WM_COMMAND, idcAdd, 0);
}
return TRUE;
case WM_HELP:
case WM_CONTEXTMENU:
return OnContextHelp(hDlg, iMsg, wParam, lParam, g_rgCtxMapMultiUserGeneral);
case WM_COMMAND:
switch(HIWORD(wParam))
{
case LBN_DBLCLK:
wParam = idcProperties;
break;
case LBN_SELCHANGE:
_ManagerUpdateButtons(hDlg);
break;
}
switch(LOWORD(wParam))
{
case IDCANCEL:
case idcClose:
case IDOK:
dFoundItem = SendDlgItemMessage(hDlg, idcStartupCombo, CB_GETCURSEL, 0, 0);
if (CB_ERR == dFoundItem)
dFoundItem = 0;
if (IsDlgButtonChecked(hDlg, idcCheckDefault))
MU_SetLoginOption(GetDlgItem(hDlg,idcStartupCombo), dFoundItem);
else
MU_SetLoginOption(GetDlgItem(hDlg,idcStartupCombo), ASK_BEFORE_LOGIN);
dFoundItem = SendDlgItemMessage(hDlg, idcDefaultCombo, CB_GETCURSEL, 0, 0);
if (CB_ERR == dFoundItem)
dFoundItem = 0;
SendDlgItemMessage(hDlg, idcUserNameList, LB_GETTEXT, dFoundItem, (LPARAM)rUserInfo.szUsername);
hr = MU_UsernameToUserId(rUserInfo.szUsername, &rUserInfo.uidUserID);
Assert(SUCCEEDED(hr));
MU_MakeDefaultUser(&rUserInfo.uidUserID);
MLEndDialogWrap(hDlg, IDOK);
return true;
case idcAdd:
ZeroMemory(&rUserInfo, sizeof(USERINFO));
if (MU_UserProperties(hDlg,&rUserInfo))
{
TCHAR szMsg[ARRAYSIZE(szRes) + CCH_IDENTITY_NAME_MAX_LENGTH];
// rebuild the username list and select the newly added one
_RememberLoginOption(GetDlgItem(hDlg,idcStartupCombo));
strcpy(sResultUsername, rUserInfo.szUsername);
_FillListBoxWithUsernames(GetDlgItem(hDlg,idcUserNameList));
_FillComboBoxWithUsernames(GetDlgItem(hDlg,idcStartupCombo), GetDlgItem(hDlg,idcUserNameList));
_FillComboBoxWithUsernames(GetDlgItem(hDlg,idcDefaultCombo), GetDlgItem(hDlg,idcUserNameList));
dwIndex = MU_GetLoginOptionIndex(GetDlgItem(hDlg,idcStartupCombo));
SendDlgItemMessage(hDlg, idcStartupCombo, CB_SETCURSEL,(dwIndex == ASK_BEFORE_LOGIN ? 0 : dwIndex) , 0);
SendDlgItemMessage(hDlg, idcDefaultCombo, CB_SETCURSEL, MU_GetDefaultOptionIndex(GetDlgItem(hDlg, idcDefaultCombo)), 0);
dFoundItem = SendDlgItemMessage(hDlg, idcUserNameList, LB_FINDSTRING, 0, (LPARAM)sResultUsername);
if (LB_ERR != dFoundItem)
{
SendDlgItemMessage(hDlg, idcUserNameList, LB_SETCURSEL, dFoundItem, 0);
}
PostMessage(HWND_BROADCAST, WM_IDENTITY_INFO_CHANGED, 0, IIC_IDENTITY_ADDED);
if (pmpParams->pszUsername)
{
MLLoadStringA(idsLoginAsUser, szRes, ARRAYSIZE(szRes));
wsprintf(szMsg, szRes, rUserInfo.szUsername);
MLLoadStringA(idsUserAdded, szRes, ARRAYSIZE(szRes));
if (IDYES == MessageBox(hDlg, szMsg, szRes, MB_YESNO))
{
lstrcpy(pmpParams->pszUsername, rUserInfo.szUsername);
PostMessage(hDlg, WM_COMMAND, idcClose, 0);
}
}
}
_ManagerUpdateButtons(hDlg);
return true;
case idcDefaultCombo:
dFoundItem = SendDlgItemMessage(hDlg, idcDefaultCombo, CB_GETCURSEL, 0, 0);
if (CB_ERR == dFoundItem)
dFoundItem = 0;
SendDlgItemMessage(hDlg, idcUserNameList, LB_GETTEXT, dFoundItem, (LPARAM)rUserInfo.szUsername);
hr = MU_UsernameToUserId(rUserInfo.szUsername, &rUserInfo.uidUserID);
Assert(SUCCEEDED(hr));
MU_MakeDefaultUser(&rUserInfo.uidUserID);
_ManagerUpdateButtons(hDlg);
break;
case idcCheckDefault:
EnableWindow(GetDlgItem(hDlg, idcStartupCombo), IsDlgButtonChecked(hDlg, idcCheckDefault));
return true;
case idcDelete:
dFoundItem = SendDlgItemMessage(hDlg, idcUserNameList, LB_GETCURSEL, 0, 0);
SendDlgItemMessage(hDlg, idcUserNameList, LB_GETTEXT, dFoundItem, (LPARAM)rUserInfo.szUsername);
hr = MU_UsernameToUserId(rUserInfo.szUsername, &rUserInfo.uidUserID);
Assert(SUCCEEDED(hr));
if (MU_ConfirmDeleteUser(hDlg, rUserInfo.szUsername))
{
MU_DeleteUser(&rUserInfo.uidUserID);
_RememberLoginOption(GetDlgItem(hDlg,idcStartupCombo));
_FillListBoxWithUsernames(GetDlgItem(hDlg,idcUserNameList));
_FillComboBoxWithUsernames(GetDlgItem(hDlg,idcStartupCombo), GetDlgItem(hDlg,idcUserNameList));
_FillComboBoxWithUsernames(GetDlgItem(hDlg,idcDefaultCombo), GetDlgItem(hDlg,idcUserNameList));
dwIndex = MU_GetLoginOptionIndex(GetDlgItem(hDlg,idcStartupCombo));
SendDlgItemMessage(hDlg, idcStartupCombo, CB_SETCURSEL,(dwIndex == ASK_BEFORE_LOGIN ? 0 : dwIndex) , 0);
SendDlgItemMessage(hDlg, idcDefaultCombo, CB_SETCURSEL, MU_GetDefaultOptionIndex(GetDlgItem(hDlg, idcDefaultCombo)), 0);
_ManagerUpdateButtons(hDlg);
}
return true;
case idcProperties:
dFoundItem = SendDlgItemMessage(hDlg, idcUserNameList, LB_GETCURSEL, 0, 0);
SendDlgItemMessage(hDlg, idcUserNameList, LB_GETTEXT, dFoundItem, (LPARAM)rUserInfo.szUsername);
hr = MU_UsernameToUserId(rUserInfo.szUsername, &rUserInfo.uidUserID);
Assert(SUCCEEDED(hr));
#ifdef IDENTITY_PASSWORDS
if (SUCCEEDED(hr) && MU_GetUserInfo(&rUserInfo.uidUserID, &rUserInfo) && MU_CanEditIdentity(hDlg, &rUserInfo.uidUserID))
#else
if (SUCCEEDED(hr) && MU_GetUserInfo(&rUserInfo.uidUserID, &rUserInfo))
#endif //IDENTITY_PASSWORDS
{
if (MU_UserProperties(hDlg,&rUserInfo))
{
// rebuild the username list and select the newly added one
_RememberLoginOption(GetDlgItem(hDlg,idcStartupCombo));
strcpy(sResultUsername, rUserInfo.szUsername);
_FillListBoxWithUsernames(GetDlgItem(hDlg,idcUserNameList));
_FillComboBoxWithUsernames(GetDlgItem(hDlg,idcStartupCombo), GetDlgItem(hDlg,idcUserNameList));
_FillComboBoxWithUsernames(GetDlgItem(hDlg,idcDefaultCombo), GetDlgItem(hDlg,idcUserNameList));
dwIndex = MU_GetLoginOptionIndex(GetDlgItem(hDlg,idcStartupCombo));
SendDlgItemMessage(hDlg, idcStartupCombo, CB_SETCURSEL,(dwIndex == ASK_BEFORE_LOGIN ? 0 : dwIndex) , 0);
SendDlgItemMessage(hDlg, idcDefaultCombo, CB_SETCURSEL, MU_GetDefaultOptionIndex(GetDlgItem(hDlg, idcDefaultCombo)), 0);
dFoundItem = SendDlgItemMessage(hDlg, idcUserNameList, LB_FINDSTRING, 0, (LPARAM)sResultUsername);
if (LB_ERR != dFoundItem)
{
SendDlgItemMessage(hDlg, idcUserNameList, LB_SETCURSEL, dFoundItem, 0);
}
}
}
_ManagerUpdateButtons(hDlg);
break;
/*
case idcDefault:
dFoundItem = SendDlgItemMessage(hDlg, idcUserNameList, LB_GETCURSEL, 0, 0);
SendDlgItemMessage(hDlg, idcUserNameList, LB_GETTEXT, dFoundItem, (LPARAM)rUserInfo.szUsername);
// _StripDefault(rUserInfo.szUsername);
hr = MU_UsernameToUserId(rUserInfo.szUsername, &rUserInfo.uidUserID);
Assert(SUCCEEDED(hr));
MU_MakeDefaultUser(&rUserInfo.uidUserID);
_RememberLoginOption(GetDlgItem(hDlg,idcStartupCombo));
_FillListBoxWithUsernames(GetDlgItem(hDlg,idcUserNameList));
_FillComboBoxWithUsernames(GetDlgItem(hDlg,idcStartupCombo), GetDlgItem(hDlg,idcUserNameList));
SendDlgItemMessage(hDlg, idcStartupCombo, CB_SETCURSEL, MU_GetLoginOptionIndex(GetDlgItem(hDlg,idcStartupCombo)), 0);
SendDlgItemMessage(hDlg, idcUserNameList, LB_SETCURSEL, dFoundItem, 0);
_ManagerUpdateButtons(hDlg);
break;
*/
}
break;
}
return false;
}
/*
MU_ManageUsers
*/
BOOL MU_ManageUsers(HWND hwnd, TCHAR *lpszSwitchtoUsername, DWORD dwFlags)
{
INT_PTR bResult;
MANAGE_PARAMS rParams;
INITCOMMONCONTROLSEX icex;
Assert(hwnd);
Assert(lpszUsername);
// make sure ICC_NATIVEFNTCTL_CLASS is inited
icex.dwSize = sizeof(INITCOMMONCONTROLSEX);
icex.dwICC = ICC_NATIVEFNTCTL_CLASS;
InitCommonControlsEx(&icex);
rParams.dwFlags = dwFlags;
rParams.pszUsername = lpszSwitchtoUsername;
bResult = MLDialogBoxParamWrap(MLGetHinst(), MAKEINTRESOURCEW(iddManager), hwnd, _ManagerDlgProc, (LPARAM)&rParams);
return (IDOK == bResult);
}