283 lines
7.8 KiB
C++
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
|