windows-nt/Source/XPSP1/NT/shell/cpls/appwzdui/setupenum.cpp
2020-09-26 16:20:57 +08:00

283 lines
7.8 KiB
C++

//---------------------------------------------------------------------------
//
// Copyright (c) Microsoft Corporation
//
// File: setupenum.cpp
//
// The current order of enumeration is whatever order we read from the registry
//
// History:
// 6-11-98 by toddb
//------------------------------------------------------------------------
#include "priv.h"
// Do not build this file if on Win9X or NT4
#ifndef DOWNLEVEL_PLATFORM
#include <shellp.h> // for IsUserAnAdmin
#include "setupenum.h"
#include "appwizid.h"
#define c_szOCSetupKey TEXT("Software\\Microsoft\\Windows NT\\CurrentVersion\\Setup\\OCManager\\ToDoList")
//-----------------------------------------------------------------------
// OCSetupApp
//-----------------------------------------------------------------------
COCSetupApp::COCSetupApp()
{
// This must be heap alloced so everything should be zero'ed out.
// Make sure this wasn't stack alloced using these asserts:
ASSERT(0 == _szDisplayName[0]);
ASSERT(0 == _szApp[0]);
ASSERT(0 == _szArgs[0]);
}
COCSetupApp::~COCSetupApp()
{
}
//-----------------------------------------------------------------------
// GetAppInfo
//
// Fills in the only valid field in our psuedo APPINFODATA structure.
BOOL COCSetupApp::GetAppInfo(PAPPINFODATA pai)
{
if (pai->cbSize != SIZEOF(APPINFODATA))
return FALSE;
DWORD dwInfoFlags = pai->dwMask;
pai->dwMask = 0;
if (dwInfoFlags & AIM_DISPLAYNAME)
{
if (SUCCEEDED(SHStrDup(_szDisplayName, &pai->pszDisplayName)))
pai->dwMask |= AIM_DISPLAYNAME;
}
return TRUE;
}
//-----------------------------------------------------------------------
// ReadFromKey
//
// This function reads the actual data from the given reg key. It returns
// TRUE if all required fields contained string data.
BOOL COCSetupApp::ReadFromKey( HKEY hkey )
{
DWORD dwType;
DWORD dwSize;
dwSize = sizeof(_szDisplayName);
if ( ERROR_SUCCESS != RegQueryValueEx( hkey, TEXT("Title"), 0, &dwType, (LPBYTE)_szDisplayName, &dwSize ) ||
dwType != REG_SZ )
{
// DisplayName is required
return FALSE;
}
dwSize = sizeof(_szApp);
if ( ERROR_SUCCESS == RegQueryValueEx( hkey, TEXT("ConfigCommand"), 0, &dwType, (LPBYTE)_szApp, &dwSize ) &&
(dwType == REG_SZ || dwType == REG_EXPAND_SZ) )
{
if ( dwType == REG_EXPAND_SZ )
{
TCHAR szBuf[MAX_PATH];
ExpandEnvironmentStrings(_szApp, szBuf, ARRAYSIZE(szBuf));
lstrcpyn(_szApp, szBuf, ARRAYSIZE(_szApp));
}
}
else
{
// ConfigCommand is required
return FALSE;
}
dwSize = sizeof(_szArgs);
if ( ERROR_SUCCESS == RegQueryValueEx( hkey, TEXT("ConfigArgs"), 0, &dwType, (LPBYTE)_szArgs, &dwSize ) &&
(dwType == REG_SZ || dwType == REG_EXPAND_SZ) )
{
if ( dwType == REG_EXPAND_SZ )
{
TCHAR szBuf[MAX_PATH];
ExpandEnvironmentStrings(_szArgs, szBuf, ARRAYSIZE(szBuf));
lstrcpyn(_szArgs, szBuf, ARRAYSIZE(_szArgs));
}
}
else
{
// This is optional so we don't fail. Instead simply insure that _szArgs is an empty string.
_szArgs[0] = 0;
}
return TRUE;
}
BOOL COCSetupApp::Run()
{
// REARCHITECT: (stephstm, 03/17/99) we should probably wait on a job object in case
// the spawned process spawns some other process(es) and then exits before them.
BOOL fRet = FALSE;
SHELLEXECUTEINFO sei = {0};
sei.cbSize = sizeof(SHELLEXECUTEINFO);
sei.fMask = SEE_MASK_NOCLOSEPROCESS;
sei.hwnd = GetDesktopWindow();
sei.lpFile = _szApp;
sei.lpParameters = _szArgs[0] ? _szArgs : NULL;
sei.nShow = SW_SHOWDEFAULT;
fRet = ShellExecuteEx(&sei);
if (fRet)
{
DWORD dwRet;
do
{
MSG msg;
// Get and process the messages!
while (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
// MsgWaitForMultipleObjects can fail with -1 being returned!
dwRet = MsgWaitForMultipleObjects(1, &sei.hProcess, FALSE, INFINITE, QS_ALLINPUT);
}
while ((WAIT_OBJECT_0 != dwRet) && (-1 != dwRet));
// Did MsgWait... failed?
if (-1 == dwRet)
{
// Yes, kill the process
TerminateProcess(sei.hProcess, 0);
fRet = FALSE;
}
CloseHandle(sei.hProcess);
}
else
{
ShellMessageBox(HINST_THISDLL, sei.hwnd, MAKEINTRESOURCE( IDS_CONFIGURE_FAILED ),
MAKEINTRESOURCE( IDS_NAME ),
MB_OK | MB_ICONEXCLAMATION);
}
return fRet;
}
//-----------------------------------------------------------------------
// OCSetupEnum
//-----------------------------------------------------------------------
COCSetupEnum::COCSetupEnum()
{
_hkeyRoot = 0;
_iRegEnumIndex = -1;
}
COCSetupEnum::~COCSetupEnum()
{
if ( _hkeyRoot )
{
RegCloseKey( _hkeyRoot );
}
}
//-----------------------------------------------------------------------
// s_OCSetupNeeded
//
// This checks for the neccessaary conditions to display the OC Setup portion of the ARP.
// This section is only shown if the current user is a member of the administrators group
// AND there are any items listed in the registry that need to be displayed.
BOOL COCSetupEnum::s_OCSetupNeeded()
{
BOOL fResult = FALSE;
HKEY hkey;
// Temporarily open the reg key to see if it exists and has any sub keys
if ( ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE, c_szOCSetupKey, 0, KEY_READ, &hkey ) )
{
TCHAR szBuf[MAX_PATH];
if ( ERROR_SUCCESS == RegEnumKey( hkey, 0, szBuf, ARRAYSIZE(szBuf) ) )
{
// Yes, there are OCSetup items, but is the current user an administrator?
if ( IsUserAnAdmin() )
{
fResult = TRUE;
}
}
RegCloseKey( hkey );
}
return fResult;
}
//-----------------------------------------------------------------------
// EnumOCSetupItems
//
// This begins the enumeration by opening the required registry key. This does
// not attempt to read any of the sub items so there is no garentee that the
// first call to Next() will succeed.
BOOL COCSetupEnum::EnumOCSetupItems()
{
ASSERT( NULL == _hkeyRoot );
// Open the reg key, return true if it's open. We leave the key open until
// our destructor is called since we need this key to do the enumeration.
if ( ERROR_SUCCESS == RegOpenKeyEx(
HKEY_LOCAL_MACHINE,
c_szOCSetupKey,
0,
KEY_READ,
&_hkeyRoot ) )
{
return TRUE;
}
return FALSE;
}
//-----------------------------------------------------------------------
// Next
//
// Reads the data from the next sub key of _hkeyRoot and returns the data in the
// out pointer. Returns TRUE if the out pointer is a valid COCSetupApp object.
BOOL COCSetupEnum::Next(COCSetupApp **ppocsa)
{
HKEY hkeySub;
TCHAR szSubKeyName[MAX_PATH];
// We open each subkey of the root key and attempt to read an OCSetup item from the subkey.
if ( ERROR_SUCCESS == RegEnumKey( _hkeyRoot, ++_iRegEnumIndex, szSubKeyName, ARRAYSIZE(szSubKeyName) ) )
{
if ( ERROR_SUCCESS == RegOpenKeyEx( _hkeyRoot, szSubKeyName, 0, KEY_READ, &hkeySub ) )
{
*ppocsa = new COCSetupApp();
if ( *ppocsa )
{
if ( (*ppocsa)->ReadFromKey( hkeySub ) )
{
RegCloseKey( hkeySub );
return TRUE;
}
delete *ppocsa;
}
RegCloseKey( hkeySub );
}
// fall through
}
*ppocsa = NULL;
return FALSE;
}
#endif //DOWNLEVEL_PLATFORM