mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1467 lines
41 KiB
1467 lines
41 KiB
/*++
|
|
|
|
Copyright (c) 1994-2000, Microsoft Corporation All rights reserved.
|
|
|
|
Module Name:
|
|
|
|
advdlg.c
|
|
|
|
Abstract:
|
|
|
|
This module implements the advanced property sheet for the Regional
|
|
Options applet.
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
|
|
|
|
//
|
|
// Include Files.
|
|
//
|
|
|
|
#include "intl.h"
|
|
#include <windowsx.h>
|
|
#include <setupapi.h>
|
|
#include <syssetup.h>
|
|
#include "intlhlp.h"
|
|
#include "maxvals.h"
|
|
|
|
|
|
|
|
|
|
//
|
|
// Context Help Ids.
|
|
//
|
|
|
|
static int aAdvancedHelpIds[] =
|
|
{
|
|
IDC_GROUPBOX1, IDH_COMM_GROUPBOX,
|
|
IDC_GROUPBOX2, IDH_COMM_GROUPBOX,
|
|
IDC_GROUPBOX3, IDH_COMM_GROUPBOX,
|
|
IDC_CODEPAGES, IDH_INTL_ADV_CODEPAGES,
|
|
IDC_SYSTEM_LOCALE, IDH_INTL_ADV_SYSTEM_LOCALE,
|
|
IDC_SYSTEM_LOCALE_TEXT1, IDH_INTL_ADV_SYSTEM_LOCALE,
|
|
IDC_SYSTEM_LOCALE_TEXT2, IDH_INTL_ADV_SYSTEM_LOCALE,
|
|
IDC_DEFAULT_USER, IDH_INTL_ADV_CHANGE,
|
|
|
|
0, 0
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Advanced_ListViewCustomDraw
|
|
//
|
|
// Processing for a list view NM_CUSTOMDRAW notification message.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
void Advanced_ListViewCustomDraw(
|
|
HWND hDlg,
|
|
LPNMLVCUSTOMDRAW pDraw)
|
|
{
|
|
LPCODEPAGE pNode;
|
|
|
|
//
|
|
// Tell the list view to notify me of item draws.
|
|
//
|
|
if (pDraw->nmcd.dwDrawStage == CDDS_PREPAINT)
|
|
{
|
|
SetWindowLongPtr(hDlg, DWLP_MSGRESULT, CDRF_NOTIFYITEMDRAW);
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Handle the Item Prepaint.
|
|
//
|
|
pNode = (LPCODEPAGE)(pDraw->nmcd.lItemlParam);
|
|
if ((pDraw->nmcd.dwDrawStage & CDDS_ITEMPREPAINT) &&
|
|
(pNode) && (pNode != (LPCODEPAGE)(LB_ERR)))
|
|
{
|
|
if (pNode->wStatus & (ML_PERMANENT | ML_DISABLE))
|
|
{
|
|
pDraw->clrText = (pDraw->nmcd.uItemState & CDIS_SELECTED)
|
|
? ((GetSysColor(COLOR_HIGHLIGHT) ==
|
|
GetSysColor(COLOR_GRAYTEXT))
|
|
? GetSysColor(COLOR_HIGHLIGHTTEXT)
|
|
: GetSysColor(COLOR_GRAYTEXT))
|
|
: GetSysColor(COLOR_GRAYTEXT);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Do the default action.
|
|
//
|
|
SetWindowLongPtr(hDlg, DWLP_MSGRESULT, CDRF_DODEFAULT);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Advanced_ListViewChanging
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL Advanced_ListViewChanging(
|
|
HWND hDlg,
|
|
NM_LISTVIEW *pLV)
|
|
{
|
|
LPCODEPAGE pNode;
|
|
|
|
//
|
|
// Make sure it's a state change message.
|
|
//
|
|
if ((((*pLV).hdr).idFrom != IDC_CODEPAGES) ||
|
|
(!(pLV->uChanged & LVIF_STATE)) ||
|
|
((pLV->uNewState & 0x3000) == 0))
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Get the item data for the currently selected item.
|
|
//
|
|
pNode = (LPCODEPAGE)(pLV->lParam);
|
|
|
|
//
|
|
// Make sure we're not trying to change a permanent or disabled
|
|
// code page. If so, return TRUE to prevent the change.
|
|
//
|
|
if ((pNode) && (pNode->wStatus & (ML_PERMANENT | ML_DISABLE)))
|
|
{
|
|
return (TRUE);
|
|
}
|
|
|
|
//
|
|
// Return FALSE to allow the change.
|
|
//
|
|
return (FALSE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Advanced_ListViewChanged
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL Advanced_ListViewChanged(
|
|
HWND hDlg,
|
|
int iID,
|
|
NM_LISTVIEW *pLV)
|
|
{
|
|
HWND hwndLV = GetDlgItem(hDlg, iID);
|
|
LPCODEPAGE pNode;
|
|
BOOL bChecked;
|
|
int iCount;
|
|
|
|
//
|
|
// Make sure it's a state change message.
|
|
//
|
|
if ((((*pLV).hdr).idFrom != IDC_CODEPAGES) ||
|
|
(!(pLV->uChanged & LVIF_STATE)) ||
|
|
((pLV->uNewState & 0x3000) == 0))
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Get the state of the check box for the currently selected item.
|
|
//
|
|
bChecked = ListView_GetCheckState(hwndLV, pLV->iItem) ? TRUE : FALSE;
|
|
|
|
//
|
|
// Get the item data for the currently selected item.
|
|
//
|
|
pNode = (LPCODEPAGE)(pLV->lParam);
|
|
|
|
//
|
|
// Make sure we're not trying to change a permanent or disabled
|
|
// code page. If so, set the check box to its appropriate state.
|
|
//
|
|
if (pNode->wStatus & (ML_PERMANENT | ML_DISABLE))
|
|
{
|
|
if (pNode->wStatus & ML_PERMANENT)
|
|
{
|
|
if (bChecked == FALSE)
|
|
{
|
|
ListView_SetCheckState(hwndLV, pLV->iItem, TRUE);
|
|
}
|
|
}
|
|
else // ML_DISABLE only
|
|
{
|
|
if ((bChecked == FALSE) && (pNode->wStatus & ML_ORIG_INSTALLED))
|
|
{
|
|
ListView_SetCheckState(hwndLV, pLV->iItem, TRUE);
|
|
}
|
|
else if ((bChecked == TRUE) && (!(pNode->wStatus & ML_ORIG_INSTALLED)))
|
|
{
|
|
ListView_SetCheckState(hwndLV, pLV->iItem, FALSE);
|
|
}
|
|
}
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Store the proper info in the code page structure.
|
|
//
|
|
pNode->wStatus &= (ML_ORIG_INSTALLED | ML_STATIC);
|
|
pNode->wStatus |= ((bChecked) ? ML_INSTALL : ML_REMOVE);
|
|
|
|
//
|
|
// Deselect all items.
|
|
//
|
|
iCount = ListView_GetItemCount(hwndLV);
|
|
while (iCount > 0)
|
|
{
|
|
ListView_SetItemState( hwndLV,
|
|
iCount - 1,
|
|
0,
|
|
LVIS_FOCUSED | LVIS_SELECTED );
|
|
iCount--;
|
|
}
|
|
|
|
//
|
|
// Make sure this item is selected.
|
|
//
|
|
ListView_SetItemState( hwndLV,
|
|
pLV->iItem,
|
|
LVIS_FOCUSED | LVIS_SELECTED,
|
|
LVIS_FOCUSED | LVIS_SELECTED );
|
|
|
|
//
|
|
// Return success.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Advanced_ListViewClick
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL Advanced_ListViewClick(
|
|
HWND hDlg,
|
|
LPNMHDR lpNmHdr)
|
|
{
|
|
LV_HITTESTINFO ht;
|
|
HWND hwndList = GetDlgItem(hDlg, IDC_CODEPAGES);
|
|
|
|
//
|
|
// Remove unnecessary processing.
|
|
//
|
|
if (lpNmHdr->idFrom != IDC_CODEPAGES)
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Get where we were hit and then translate it to our
|
|
// window.
|
|
//
|
|
GetCursorPos(&ht.pt);
|
|
ScreenToClient(hwndList, &ht.pt);
|
|
ListView_HitTest(hwndList, &ht);
|
|
if ((ht.iItem >= 0) && ((ht.flags & LVHT_ONITEM) == LVHT_ONITEMLABEL))
|
|
{
|
|
UINT state;
|
|
|
|
//
|
|
// The user clicked on the item label. Simulate a
|
|
// state change so we can process it.
|
|
//
|
|
state = ListView_GetItemState( hwndList,
|
|
ht.iItem,
|
|
LVIS_STATEIMAGEMASK );
|
|
state ^= INDEXTOSTATEIMAGEMASK(LVIS_SELECTED | LVIS_FOCUSED);
|
|
|
|
//
|
|
// The state is either selected or focused. Flip the
|
|
// bits. The SetItemState causes the system to bounce
|
|
// back a notification for LVN_ITEMCHANGED and the
|
|
// code then does the right thing. Note -- we MUST
|
|
// check for LVHT_ONITEMLABEL. If we do this code for
|
|
// LVHT_ONITEMSTATEICON, the code will get 2
|
|
// ITEMCHANGED notifications, and the state will stay
|
|
// where it is, which is not good. If we want this
|
|
// to also fire if the guy clicks in the empty space
|
|
// right of the label text, we need to look for
|
|
// LVHT_ONITEM as well as LVHT_ONITEMLABEL.
|
|
//
|
|
ListView_SetItemState( hwndList,
|
|
ht.iItem,
|
|
state,
|
|
LVIS_STATEIMAGEMASK );
|
|
}
|
|
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Advanced_GetSupportedCodePages
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL Advanced_GetSupportedCodePages()
|
|
{
|
|
UINT CodePage;
|
|
HANDLE hCodePage;
|
|
LPCODEPAGE pCP;
|
|
INFCONTEXT Context;
|
|
TCHAR szSection[MAX_PATH];
|
|
int LineCount, LineNum;
|
|
CPINFOEX Info;
|
|
|
|
//
|
|
// Get the number of supported code pages from the inf file.
|
|
//
|
|
LineCount = (UINT)SetupGetLineCount(g_hIntlInf, TEXT("CodePages"));
|
|
if (LineCount <= 0)
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Go through all supported code pages in the inf file.
|
|
//
|
|
for (LineNum = 0; LineNum < LineCount; LineNum++)
|
|
{
|
|
if (SetupGetLineByIndex(g_hIntlInf, TEXT("CodePages"), LineNum, &Context) &&
|
|
SetupGetIntField(&Context, 0, &CodePage))
|
|
{
|
|
//
|
|
// Create the new node.
|
|
//
|
|
if (!(hCodePage = GlobalAlloc(GHND, sizeof(CODEPAGE))))
|
|
{
|
|
return (FALSE);
|
|
}
|
|
pCP = GlobalLock(hCodePage);
|
|
|
|
//
|
|
// Fill in the new node with the appropriate info.
|
|
//
|
|
pCP->wStatus = 0;
|
|
pCP->CodePage = CodePage;
|
|
pCP->hCodePage = hCodePage;
|
|
(pCP->pszName)[0] = 0;
|
|
|
|
//
|
|
// Get the appropriate display string.
|
|
//
|
|
if (GetCPInfoEx(CodePage, 0, &Info))
|
|
{
|
|
lstrcpy(pCP->pszName, Info.CodePageName);
|
|
}
|
|
else if (!SetupGetStringField(&Context, 1, pCP->pszName, MAX_PATH, NULL))
|
|
{
|
|
GlobalUnlock(hCodePage);
|
|
GlobalFree(hCodePage);
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// See if this code page can be removed.
|
|
//
|
|
wsprintf(szSection, TEXT("%ws%d"), szCPRemovePrefix, CodePage);
|
|
if ((CodePage == GetACP()) ||
|
|
(CodePage == GetOEMCP()) ||
|
|
(!SetupFindFirstLine( g_hIntlInf,
|
|
szSection,
|
|
TEXT("AddReg"),
|
|
&Context )))
|
|
{
|
|
//
|
|
// Mark it as permanent.
|
|
// Also mark it as originally installed to avoid problems.
|
|
//
|
|
pCP->wStatus |= (ML_ORIG_INSTALLED | ML_PERMANENT);
|
|
}
|
|
|
|
//
|
|
// Add the code page to the front of the linked list.
|
|
//
|
|
pCP->pNext = pCodePages;
|
|
pCodePages = pCP;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Return success.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Advanced_InitSystemLocales
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL Advanced_InitSystemLocales(
|
|
HWND hDlg)
|
|
{
|
|
TCHAR szSystemBuf[SIZE_128];
|
|
TCHAR szDefaultSystemBuf[SIZE_128];
|
|
TCHAR szBuf[SIZE_128];
|
|
DWORD dwIndex;
|
|
HWND hSystemLocale = GetDlgItem(hDlg, IDC_SYSTEM_LOCALE);
|
|
|
|
//
|
|
// Get the list of locales and fill in the system locale
|
|
// combo box.
|
|
//
|
|
Intl_EnumLocales(hDlg, hSystemLocale, TRUE);
|
|
|
|
//
|
|
// Get the string for the system default setting.
|
|
// Special case Spanish.
|
|
//
|
|
if ((SysLocaleID == LCID_SPANISH_TRADITIONAL) ||
|
|
(SysLocaleID == LCID_SPANISH_INTL))
|
|
{
|
|
LoadString(hInstance, IDS_SPANISH_NAME, szSystemBuf, SIZE_128);
|
|
}
|
|
else
|
|
{
|
|
GetLocaleInfo(SysLocaleID, LOCALE_SLANGUAGE, szSystemBuf, SIZE_128);
|
|
}
|
|
|
|
//
|
|
// Select the current system default locale id in the list.
|
|
//
|
|
dwIndex = ComboBox_FindStringExact(hSystemLocale, -1, szSystemBuf);
|
|
if (dwIndex == CB_ERR)
|
|
{
|
|
dwIndex = ComboBox_FindStringExact(hSystemLocale, -1, szDefaultSystemBuf);
|
|
if (dwIndex == CB_ERR)
|
|
{
|
|
GetLocaleInfo(US_LOCALE, LOCALE_SLANGUAGE, szBuf, SIZE_128);
|
|
dwIndex = ComboBox_FindStringExact(hSystemLocale, -1, szBuf);
|
|
if (dwIndex == CB_ERR)
|
|
{
|
|
dwIndex = 0;
|
|
}
|
|
}
|
|
}
|
|
ComboBox_SetCurSel(hSystemLocale, dwIndex);
|
|
|
|
//
|
|
// Return success.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Advanced_SetSystemLocale
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL Advanced_SetSystemLocale(
|
|
HWND hDlg)
|
|
{
|
|
HWND hSystemLocale = GetDlgItem(hDlg, IDC_SYSTEM_LOCALE);
|
|
DWORD dwLocale;
|
|
LCID NewLocale;
|
|
HCURSOR hcurSave;
|
|
|
|
//
|
|
// Put up the hour glass.
|
|
//
|
|
hcurSave = SetCursor(LoadCursor(NULL, IDC_WAIT));
|
|
|
|
//
|
|
// Get the current selection.
|
|
//
|
|
dwLocale = ComboBox_GetCurSel(hSystemLocale);
|
|
|
|
//
|
|
// Get the locale id for the current selection and save it.
|
|
//
|
|
NewLocale = (LCID)ComboBox_GetItemData(hSystemLocale, dwLocale);
|
|
if (IsValidLocale(NewLocale, LCID_SUPPORTED))
|
|
{
|
|
SysLocaleID = NewLocale;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// This shouldn't happen, since the values in the combo box
|
|
// should already be installed via the language groups.
|
|
// Put up an error message just in case.
|
|
//
|
|
SetCursor(hcurSave);
|
|
ShowMsg( NULL,
|
|
IDS_SETUP_STRING,
|
|
IDS_TITLE_STRING,
|
|
MB_OK_OOPS,
|
|
NULL );
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Turn off the hour glass.
|
|
//
|
|
SetCursor(hcurSave);
|
|
|
|
//
|
|
// Return success.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Advanced_InitCodePages
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL Advanced_InitCodePages(
|
|
HWND hDlg,
|
|
BOOL bInitTime)
|
|
{
|
|
HWND hwndCP = GetDlgItem(hDlg, IDC_CODEPAGES);
|
|
LPCODEPAGE pCP;
|
|
DWORD dwExStyle;
|
|
RECT Rect;
|
|
LV_COLUMN Column;
|
|
LV_ITEM Item;
|
|
int iIndex;
|
|
|
|
//
|
|
// Open the Inf file.
|
|
//
|
|
g_hIntlInf = SetupOpenInfFile(szIntlInf, NULL, INF_STYLE_WIN4, NULL);
|
|
if (g_hIntlInf == INVALID_HANDLE_VALUE)
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
if (!SetupOpenAppendInfFile(NULL, g_hIntlInf, NULL))
|
|
{
|
|
SetupCloseInfFile(g_hIntlInf);
|
|
g_hIntlInf = NULL;
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Get all supported code pages from the inf file.
|
|
//
|
|
if (Advanced_GetSupportedCodePages() == FALSE)
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Close the inf file.
|
|
//
|
|
SetupCloseInfFile(g_hIntlInf);
|
|
g_hIntlInf = NULL;
|
|
|
|
//
|
|
// Enumerate all installed code pages.
|
|
//
|
|
if (EnumSystemCodePages(Intl_EnumInstalledCPProc, CP_INSTALLED) == FALSE)
|
|
{
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// We only want to do this the first time we setup the list view.
|
|
// Otherwise, we get multiple columns created.
|
|
//
|
|
if (bInitTime)
|
|
{
|
|
//
|
|
// Create a column for the list view.
|
|
//
|
|
GetClientRect(hwndCP, &Rect);
|
|
Column.mask = LVCF_FMT | LVCF_TEXT | LVCF_WIDTH;
|
|
Column.fmt = LVCFMT_LEFT;
|
|
Column.cx = Rect.right - GetSystemMetrics(SM_CYHSCROLL);
|
|
Column.pszText = NULL;
|
|
Column.cchTextMax = 0;
|
|
Column.iSubItem = 0;
|
|
ListView_InsertColumn(hwndCP, 0, &Column);
|
|
|
|
//
|
|
// Set extended list view style to use the check boxes.
|
|
//
|
|
dwExStyle = ListView_GetExtendedListViewStyle(hwndCP);
|
|
ListView_SetExtendedListViewStyle( hwndCP,
|
|
dwExStyle |
|
|
LVS_EX_CHECKBOXES |
|
|
LVS_EX_FULLROWSELECT );
|
|
}
|
|
|
|
//
|
|
// Go through the list of code pages and add each one to the
|
|
// list view and set the appropriate state.
|
|
//
|
|
pCP = pCodePages;
|
|
while (pCP)
|
|
{
|
|
//
|
|
// Insert the item into the list view.
|
|
//
|
|
Item.mask = LVIF_TEXT | LVIF_PARAM | LVIF_STATE;
|
|
Item.iItem = 0;
|
|
Item.iSubItem = 0;
|
|
Item.state = 0;
|
|
Item.stateMask = LVIS_STATEIMAGEMASK;
|
|
Item.pszText = pCP->pszName;
|
|
Item.cchTextMax = 0;
|
|
Item.iImage = 0;
|
|
Item.lParam = (LPARAM)pCP;
|
|
|
|
iIndex = ListView_InsertItem(hwndCP, &Item);
|
|
|
|
//
|
|
// Set the checked state.
|
|
//
|
|
// There's a bug in the list view code such that the check mark
|
|
// isn't displayed when you set the state through InsertItem, so
|
|
// we have to set it explicitly using SetItemState.
|
|
//
|
|
if (iIndex >= 0)
|
|
{
|
|
ListView_SetItemState( hwndCP,
|
|
iIndex,
|
|
(pCP->wStatus & ML_ORIG_INSTALLED)
|
|
? INDEXTOSTATEIMAGEMASK(LVIS_SELECTED)
|
|
: INDEXTOSTATEIMAGEMASK(LVIS_FOCUSED),
|
|
LVIS_STATEIMAGEMASK );
|
|
}
|
|
|
|
//
|
|
// Advance to the next code page.
|
|
//
|
|
pCP = pCP->pNext;
|
|
}
|
|
|
|
//
|
|
// Deselect all items.
|
|
//
|
|
iIndex = ListView_GetItemCount(hwndCP);
|
|
while (iIndex > 0)
|
|
{
|
|
ListView_SetItemState( hwndCP,
|
|
iIndex - 1,
|
|
0,
|
|
LVIS_FOCUSED | LVIS_SELECTED );
|
|
iIndex--;
|
|
}
|
|
|
|
//
|
|
// Select the first one in the list.
|
|
//
|
|
ListView_SetItemState( hwndCP,
|
|
0,
|
|
LVIS_FOCUSED | LVIS_SELECTED,
|
|
LVIS_FOCUSED | LVIS_SELECTED );
|
|
|
|
//
|
|
// Return success.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Region_FreeGlobalInfo
|
|
//
|
|
// Processing for a WM_DESTROY message.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
void Advanced_FreeGlobalInfo()
|
|
{
|
|
LPCODEPAGE pPreCP, pCurCP;
|
|
HANDLE hAlloc;
|
|
|
|
//
|
|
// Remove Code Page info.
|
|
//
|
|
pCurCP = pCodePages;
|
|
pCodePages = NULL;
|
|
|
|
while (pCurCP)
|
|
{
|
|
pPreCP = pCurCP;
|
|
pCurCP = pPreCP->pNext;
|
|
hAlloc = pPreCP->hCodePage;
|
|
GlobalUnlock(hAlloc);
|
|
GlobalFree(hAlloc);
|
|
}
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Advanced_ClearValues
|
|
//
|
|
// Reset each of the list boxes in the advanced property sheet page.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
void Advanced_ClearValues(
|
|
HWND hDlg)
|
|
{
|
|
//
|
|
// Clear the system locale
|
|
//
|
|
ComboBox_ResetContent(GetDlgItem(hDlg, IDC_SYSTEM_LOCALE));
|
|
|
|
//
|
|
// Clear the Code Page list
|
|
//
|
|
ListView_DeleteAllItems(GetDlgItem(hDlg, IDC_CODEPAGES));
|
|
Advanced_FreeGlobalInfo();
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Advanced_SetValues
|
|
//
|
|
// Initialize all of the controls in the advanced property sheet page.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
void Advanced_SetValues(
|
|
HWND hDlg,
|
|
BOOL bInitTime)
|
|
{
|
|
//
|
|
// Init system locale list.
|
|
//
|
|
Advanced_InitSystemLocales(hDlg);
|
|
|
|
//
|
|
// Init code page list view.
|
|
//
|
|
Advanced_InitCodePages(hDlg, bInitTime);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Advanced_ApplySettings
|
|
//
|
|
// For every control that has changed (that affects the Locale settings),
|
|
// call Set_Locale_Values to update the user locale information. Notify
|
|
// the parent of changes and reset the change flag stored in the property
|
|
// sheet page structure appropriately. Redisplay the time sample if
|
|
// bRedisplay is TRUE.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL Advanced_ApplySettings(
|
|
HWND hDlg)
|
|
{
|
|
LPPROPSHEETPAGE lpPropSheet = (LPPROPSHEETPAGE)(GetWindowLongPtr(hDlg, DWLP_USER));
|
|
LPARAM Changes = lpPropSheet->lParam;
|
|
HWND hSystemLocale = GetDlgItem(hDlg, IDC_SYSTEM_LOCALE);
|
|
DWORD dwLocale;
|
|
LCID NewLocale;
|
|
HCURSOR hcurSave;
|
|
BOOL InvokeSysocmgr = FALSE;
|
|
BOOL bReboot = FALSE;
|
|
|
|
//
|
|
// Put up the hour glass.
|
|
//
|
|
hcurSave = SetCursor(LoadCursor(NULL, IDC_WAIT));
|
|
|
|
//
|
|
// See if there are any changes to the codepage conversions.
|
|
//
|
|
if (Changes & AD_CodePages)
|
|
{
|
|
LPCODEPAGE pCP;
|
|
HINF hIntlInf;
|
|
HSPFILEQ FileQueue;
|
|
PVOID QueueContext;
|
|
BOOL bInitInf = FALSE;
|
|
BOOL fAdd;
|
|
BOOL bRet = TRUE;
|
|
TCHAR szSection[MAX_PATH];
|
|
|
|
//
|
|
// Go through each code page node to see if anything needs to
|
|
// be done.
|
|
//
|
|
pCP = pCodePages;
|
|
while (pCP)
|
|
{
|
|
//
|
|
// See if any changes are necessary for this code page.
|
|
//
|
|
if ((pCP->wStatus == ML_INSTALL) ||
|
|
(pCP->wStatus == (ML_ORIG_INSTALLED | ML_REMOVE)))
|
|
{
|
|
//
|
|
// See if we're installing or removing.
|
|
//
|
|
fAdd = (pCP->wStatus == ML_INSTALL);
|
|
|
|
//
|
|
// Initialize Inf stuff.
|
|
//
|
|
if ((!bInitInf) &&
|
|
(!Intl_InitInf(hDlg, &hIntlInf, szIntlInf, &FileQueue, &QueueContext)))
|
|
{
|
|
SetCursor(hcurSave);
|
|
return (FALSE);
|
|
}
|
|
bInitInf = TRUE;
|
|
|
|
//
|
|
// Get the inf section name.
|
|
//
|
|
wsprintf( szSection,
|
|
TEXT("%ws%d"),
|
|
fAdd ? szCPInstallPrefix : szCPRemovePrefix,
|
|
pCP->CodePage );
|
|
|
|
//
|
|
// Enqueue the code page files so that they may be
|
|
// copied. This only handles the CopyFiles entries in the
|
|
// inf file.
|
|
//
|
|
if (!SetupInstallFilesFromInfSection( hIntlInf,
|
|
NULL,
|
|
FileQueue,
|
|
szSection,
|
|
pSetupSourcePath,
|
|
SP_COPY_NEWER ))
|
|
{
|
|
//
|
|
// Setup failed to find the code page.
|
|
// This shouldn't happen - the inf file is messed up.
|
|
//
|
|
ShowMsg( hDlg,
|
|
IDS_ML_COPY_FAILED,
|
|
0,
|
|
MB_OK_OOPS,
|
|
pCP->pszName );
|
|
pCP->wStatus = 0;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Go to the next code page node.
|
|
//
|
|
pCP = pCP->pNext;
|
|
}
|
|
|
|
if (bInitInf)
|
|
{
|
|
DWORD d;
|
|
|
|
//
|
|
// See if we need to install any files.
|
|
//
|
|
// d = 0: User wants new files or some files were missing;
|
|
// Must commit queue.
|
|
//
|
|
// d = 1: User wants to use existing files and queue is empty;
|
|
// Can skip committing queue.
|
|
//
|
|
// d = 2: User wants to use existing files, but del/ren queues
|
|
// not empty. Must commit queue. The copy queue will
|
|
// have been emptied, so only del/ren functions will be
|
|
// performed.
|
|
//
|
|
if ((SetupScanFileQueue( FileQueue,
|
|
SPQ_SCAN_PRUNE_COPY_QUEUE |
|
|
SPQ_SCAN_FILE_VALIDITY,
|
|
GetParent(hDlg),
|
|
NULL,
|
|
NULL,
|
|
&d )) && (d != 1))
|
|
{
|
|
//
|
|
// Copy the files in the queue.
|
|
//
|
|
if (!SetupCommitFileQueue( GetParent(hDlg),
|
|
FileQueue,
|
|
SetupDefaultQueueCallback,
|
|
QueueContext ))
|
|
{
|
|
//
|
|
// This can happen if the user hits Cancel from within
|
|
// the setup dialog.
|
|
//
|
|
ShowMsg( hDlg,
|
|
IDS_ML_SETUP_FAILED,
|
|
0,
|
|
MB_OK_OOPS,
|
|
NULL );
|
|
bRet = FALSE;
|
|
goto Advanced_CodepageConverionsSetupError;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Execute all of the other code page entries in the inf file.
|
|
//
|
|
pCP = pCodePages;
|
|
while (pCP)
|
|
{
|
|
//
|
|
// See if any changes are necessary for this code page.
|
|
//
|
|
if ((pCP->wStatus == ML_INSTALL) ||
|
|
(pCP->wStatus == (ML_ORIG_INSTALLED | ML_REMOVE)))
|
|
{
|
|
fAdd = (pCP->wStatus == ML_INSTALL);
|
|
|
|
//
|
|
// Get the inf section name.
|
|
//
|
|
wsprintf( szSection,
|
|
TEXT("%ws%d"),
|
|
fAdd ? szCPInstallPrefix : szCPRemovePrefix,
|
|
pCP->CodePage );
|
|
|
|
//
|
|
// Call setup to install other inf info for this
|
|
// code page.
|
|
//
|
|
if (!SetupInstallFromInfSection( GetParent(hDlg),
|
|
hIntlInf,
|
|
szSection,
|
|
SPINST_ALL & ~SPINST_FILES,
|
|
NULL,
|
|
pSetupSourcePath,
|
|
0,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL ))
|
|
{
|
|
//
|
|
// Setup failed.
|
|
//
|
|
// Already copied the code page file, so no need to
|
|
// change the status of the code page info here.
|
|
//
|
|
// This shouldn't happen - the inf file is messed up.
|
|
//
|
|
ShowMsg( hDlg,
|
|
IDS_ML_INSTALL_FAILED,
|
|
0,
|
|
MB_OK_OOPS,
|
|
pCP->pszName );
|
|
}
|
|
|
|
//
|
|
// Reset the status to show the new state of this
|
|
// code page.
|
|
//
|
|
pCP->wStatus &= (ML_STATIC);
|
|
if (fAdd)
|
|
{
|
|
pCP->wStatus |= ML_ORIG_INSTALLED;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Clear out wStatus and go to the next code page node.
|
|
//
|
|
pCP->wStatus &= (ML_ORIG_INSTALLED | ML_STATIC);
|
|
pCP = pCP->pNext;
|
|
}
|
|
|
|
Advanced_CodepageConverionsSetupError:
|
|
//
|
|
// Close Inf stuff.
|
|
//
|
|
Intl_CloseInf(hIntlInf, FileQueue, QueueContext);
|
|
}
|
|
|
|
//
|
|
// Check if we need a reboot
|
|
//
|
|
if (RegionalChgState & AD_SystemLocale)
|
|
{
|
|
bReboot = TRUE;
|
|
}
|
|
}
|
|
|
|
//
|
|
// See if there are any changes to the system locale.
|
|
//
|
|
if (Changes & AD_SystemLocale)
|
|
{
|
|
//
|
|
// Get the current selection.
|
|
//
|
|
dwLocale = ComboBox_GetCurSel(hSystemLocale);
|
|
|
|
//
|
|
// Get the locale id for the current selection and save it.
|
|
//
|
|
NewLocale = (LCID)ComboBox_GetItemData(hSystemLocale, dwLocale);
|
|
if (IsValidLocale(NewLocale, LCID_SUPPORTED))
|
|
{
|
|
SysLocaleID = NewLocale;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// This shouldn't happen, since the values in the combo box
|
|
// should already be installed via the language groups.
|
|
// Put up an error message just in case.
|
|
//
|
|
SetCursor(hcurSave);
|
|
ShowMsg( NULL,
|
|
IDS_SETUP_STRING,
|
|
IDS_TITLE_STRING,
|
|
MB_OK_OOPS,
|
|
NULL );
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// See if the current selection is different from the original
|
|
// selection.
|
|
//
|
|
if (RegSysLocaleID != SysLocaleID)
|
|
{
|
|
//
|
|
// Call setup to install the option.
|
|
//
|
|
if (SetupChangeLocaleEx( hDlg,
|
|
SysLocaleID,
|
|
pSetupSourcePath,
|
|
(g_bSetupCase)
|
|
? SP_INSTALL_FILES_QUIETLY
|
|
: 0,
|
|
NULL,
|
|
0 ))
|
|
{
|
|
//
|
|
// If Setup fails, put up a message.
|
|
//
|
|
SetCursor(hcurSave);
|
|
ShowMsg( NULL,
|
|
IDS_SETUP_STRING,
|
|
IDS_TITLE_STRING,
|
|
MB_OK_OOPS,
|
|
NULL );
|
|
SysLocaleID = GetSystemDefaultLCID();
|
|
return (FALSE);
|
|
}
|
|
|
|
//
|
|
// Check if we need to proceed with the Font Substitution
|
|
//
|
|
if (Intl_IsUIFontSubstitute() &&
|
|
((LANGID)LANGIDFROMLCID(SysLocaleID) == Intl_GetDotDefaultUILanguage()))
|
|
{
|
|
Intl_ApplyFontSubstitute(SysLocaleID);
|
|
}
|
|
|
|
//
|
|
// Reset the registry system locale value.
|
|
//
|
|
RegSysLocaleID = SysLocaleID;
|
|
|
|
//
|
|
// Need to make sure the proper keyboard layout is installed.
|
|
//
|
|
Intl_InstallKeyboardLayout(hDlg, SysLocaleID, 0, FALSE, FALSE, TRUE);
|
|
|
|
//
|
|
// See if we need to reboot.
|
|
//
|
|
if (SysLocaleID != GetSystemDefaultLCID())
|
|
{
|
|
bReboot = TRUE;
|
|
}
|
|
|
|
InvokeSysocmgr = TRUE;
|
|
}
|
|
}
|
|
|
|
//
|
|
// If the system locale changed and we're not running
|
|
// in gui setup, then let's invoke sysocmgr.exe.
|
|
//
|
|
if (!g_bSetupCase && InvokeSysocmgr)
|
|
{
|
|
//
|
|
// Run any necessary apps (for FSVGA/FSNEC installation).
|
|
//
|
|
Intl_RunRegApps(c_szSysocmgr);
|
|
}
|
|
|
|
//
|
|
// Reset the property page settings.
|
|
//
|
|
PropSheet_UnChanged(GetParent(hDlg), hDlg);
|
|
Changes = AD_EverChg;
|
|
|
|
//
|
|
// Turn off the hour glass.
|
|
//
|
|
SetCursor(hcurSave);
|
|
|
|
//
|
|
// See if we need to display the reboot message.
|
|
//
|
|
if ((!g_bSetupCase) && (bReboot))
|
|
{
|
|
if(RegionalChgState & Process_Languages )
|
|
{
|
|
RegionalChgState &= ~(AD_CodePages | AD_SystemLocale);
|
|
}
|
|
else
|
|
{
|
|
if (ShowMsg( hDlg,
|
|
IDS_REBOOT_STRING,
|
|
IDS_TITLE_STRING,
|
|
MB_YESNO | MB_ICONQUESTION,
|
|
NULL ) == IDYES)
|
|
{
|
|
Intl_RebootTheSystem();
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Return success.
|
|
//
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Advanced_ValidatePPS
|
|
//
|
|
// Validate each of the combo boxes whose values are constrained.
|
|
// If any of the input fails, notify the user and then return FALSE
|
|
// to indicate validation failure.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL Advanced_ValidatePPS(
|
|
HWND hDlg,
|
|
LPARAM Changes)
|
|
{
|
|
//
|
|
// If nothing has changed, return TRUE immediately.
|
|
//
|
|
if (Changes <= AD_EverChg)
|
|
{
|
|
return (TRUE);
|
|
}
|
|
|
|
//
|
|
// See if the system locale has changed.
|
|
//
|
|
if (Changes & AD_SystemLocale)
|
|
{
|
|
HWND hSystemLocale = GetDlgItem(hDlg, IDC_SYSTEM_LOCALE);
|
|
DWORD dwLocale = ComboBox_GetCurSel(hSystemLocale);
|
|
LCID NewLocale;
|
|
|
|
//
|
|
// Get the locale id for the current selection and save it.
|
|
//
|
|
NewLocale = (LCID)ComboBox_GetItemData(hSystemLocale, dwLocale);
|
|
if (IsValidLocale(NewLocale, LCID_SUPPORTED))
|
|
{
|
|
SysLocaleID = NewLocale;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// This shouldn't happen, since the values in the combo box
|
|
// should already be installed via the language groups.
|
|
// Put up an error message just in case.
|
|
//
|
|
ShowMsg( NULL,
|
|
IDS_SETUP_STRING,
|
|
IDS_TITLE_STRING,
|
|
MB_OK_OOPS,
|
|
NULL );
|
|
return (FALSE);
|
|
}
|
|
}
|
|
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Advanced_InitPropSheet
|
|
//
|
|
// The extra long value for the property sheet page is used as a set of
|
|
// state or change flags for each of the list boxes in the property sheet.
|
|
// Initialize this value to 0. Call Advanced_SetValues with the property
|
|
// sheet handle to initialize all of the property sheet controls. Limit
|
|
// the length of the text in some of the ComboBoxes.
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
void Advanced_InitPropSheet(
|
|
HWND hDlg,
|
|
LPARAM lParam)
|
|
{
|
|
//
|
|
// The lParam holds a pointer to the property sheet page. Save it
|
|
// for later reference.
|
|
//
|
|
SetWindowLongPtr(hDlg, DWLP_USER, lParam);
|
|
|
|
//
|
|
// Set values.
|
|
//
|
|
if (pLanguageGroups == NULL)
|
|
{
|
|
Intl_LoadLanguageGroups(hDlg);
|
|
}
|
|
Advanced_SetValues(hDlg, TRUE);
|
|
|
|
//
|
|
// If we are in setup mode, we need to disable the Default User
|
|
// Account UI.
|
|
//
|
|
if (g_bSetupCase)
|
|
{
|
|
HWND hUIDefUserBox = GetDlgItem(hDlg, IDC_GROUPBOX3);
|
|
HWND hUIDefUser = GetDlgItem(hDlg, IDC_DEFAULT_USER);
|
|
|
|
EnableWindow(hUIDefUserBox, FALSE);
|
|
EnableWindow(hUIDefUser, FALSE);
|
|
ShowWindow(hUIDefUserBox, SW_HIDE);
|
|
ShowWindow(hUIDefUser, SW_HIDE);
|
|
}
|
|
}
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// AdvancedDlgProc
|
|
//
|
|
////////////////////////////////////////////////////////////////////////////
|
|
|
|
INT_PTR CALLBACK AdvancedDlgProc(
|
|
HWND hDlg,
|
|
UINT message,
|
|
WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
LPPROPSHEETPAGE lpPropSheet = (LPPROPSHEETPAGE)(GetWindowLongPtr(hDlg, DWLP_USER));
|
|
|
|
switch (message)
|
|
{
|
|
case ( WM_NOTIFY ) :
|
|
{
|
|
LPNMHDR lpnm = (NMHDR *)lParam;
|
|
switch (lpnm->code)
|
|
{
|
|
case ( PSN_SETACTIVE ) :
|
|
{
|
|
//
|
|
// If there has been a change in the regional Locale
|
|
// setting, clear all of the current info in the
|
|
// property sheet, get the new values, and update the
|
|
// appropriate registry values.
|
|
//
|
|
if (Verified_Regional_Chg & Process_Advanced)
|
|
{
|
|
Verified_Regional_Chg &= ~Process_Advanced;
|
|
Advanced_ClearValues(hDlg);
|
|
Advanced_SetValues(hDlg, FALSE);
|
|
lpPropSheet->lParam = 0;
|
|
}
|
|
break;
|
|
}
|
|
case ( PSN_KILLACTIVE ) :
|
|
{
|
|
//
|
|
// Validate the entries on the property page.
|
|
//
|
|
SetWindowLongPtr( hDlg,
|
|
DWLP_MSGRESULT,
|
|
!Advanced_ValidatePPS(hDlg, lpPropSheet->lParam) );
|
|
break;
|
|
}
|
|
case ( PSN_APPLY ) :
|
|
{
|
|
//
|
|
// Apply the settings.
|
|
//
|
|
if (Advanced_ApplySettings(hDlg))
|
|
{
|
|
SetWindowLongPtr( hDlg,
|
|
DWLP_MSGRESULT,
|
|
PSNRET_NOERROR );
|
|
|
|
//
|
|
// Zero out the AD_EverChg bit.
|
|
//
|
|
lpPropSheet->lParam = 0;
|
|
}
|
|
else
|
|
{
|
|
SetWindowLongPtr( hDlg,
|
|
DWLP_MSGRESULT,
|
|
PSNRET_INVALID_NOCHANGEPAGE );
|
|
}
|
|
|
|
|
|
break;
|
|
}
|
|
case ( NM_CUSTOMDRAW ) :
|
|
{
|
|
Advanced_ListViewCustomDraw(hDlg, (LPNMLVCUSTOMDRAW)lParam);
|
|
return (TRUE);
|
|
}
|
|
case ( LVN_ITEMCHANGING ) :
|
|
{
|
|
Advanced_ListViewChanging(hDlg, (NM_LISTVIEW *)lParam);
|
|
break;
|
|
}
|
|
case ( LVN_ITEMCHANGED ) :
|
|
{
|
|
//
|
|
// Save the change to the code pages.
|
|
//
|
|
if (Advanced_ListViewChanged( hDlg,
|
|
IDC_CODEPAGES,
|
|
(NM_LISTVIEW *)lParam ))
|
|
{
|
|
//
|
|
// Note that the code pages have changed and
|
|
// enable the apply button.
|
|
//
|
|
lpPropSheet->lParam |= AD_CodePages;
|
|
PropSheet_Changed(GetParent(hDlg), hDlg);
|
|
RegionalChgState |= AD_CodePages;
|
|
}
|
|
break;
|
|
}
|
|
case ( NM_CLICK ) :
|
|
case ( NM_DBLCLK ) :
|
|
{
|
|
Advanced_ListViewClick(hDlg, (NMHDR*)lParam);
|
|
}
|
|
default :
|
|
{
|
|
return (FALSE);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case ( WM_INITDIALOG ) :
|
|
{
|
|
Advanced_InitPropSheet(hDlg, lParam);
|
|
break;
|
|
}
|
|
case ( WM_DESTROY ) :
|
|
{
|
|
Advanced_FreeGlobalInfo();
|
|
break;
|
|
}
|
|
case ( WM_HELP ) :
|
|
{
|
|
WinHelp( (HWND)((LPHELPINFO)lParam)->hItemHandle,
|
|
szHelpFile,
|
|
HELP_WM_HELP,
|
|
(DWORD_PTR)(LPTSTR)aAdvancedHelpIds );
|
|
break;
|
|
}
|
|
case ( WM_CONTEXTMENU ) : // right mouse click
|
|
{
|
|
WinHelp( (HWND)wParam,
|
|
szHelpFile,
|
|
HELP_CONTEXTMENU,
|
|
(DWORD_PTR)(LPTSTR)aAdvancedHelpIds );
|
|
break;
|
|
}
|
|
case ( WM_COMMAND ) :
|
|
{
|
|
switch (LOWORD(wParam))
|
|
{
|
|
case ( IDC_SYSTEM_LOCALE ) :
|
|
{
|
|
if (HIWORD(wParam) == CBN_SELCHANGE)
|
|
{
|
|
//
|
|
// Set the AD_SystemLocale change flag.
|
|
//
|
|
lpPropSheet->lParam |= AD_SystemLocale;
|
|
PropSheet_Changed(GetParent(hDlg), hDlg);
|
|
RegionalChgState |= AD_SystemLocale;
|
|
}
|
|
break;
|
|
}
|
|
case ( IDC_DEFAULT_USER ) :
|
|
{
|
|
BOOL curState;
|
|
|
|
//
|
|
// Verify the check box state.
|
|
//
|
|
if (IsDlgButtonChecked(hDlg, IDC_DEFAULT_USER))
|
|
{
|
|
ShowMsg( hDlg,
|
|
IDS_DEF_USER_CONF,
|
|
IDS_DEF_USER_CONF_TITLE,
|
|
MB_OK_OOPS,
|
|
NULL);
|
|
|
|
g_bDefaultUser = TRUE;
|
|
}
|
|
else
|
|
{
|
|
g_bDefaultUser = FALSE;
|
|
}
|
|
|
|
//
|
|
// Set the AD_DefaultUser change flag.
|
|
//
|
|
if (g_bDefaultUser)
|
|
{
|
|
lpPropSheet->lParam |= AD_DefaultUser;
|
|
PropSheet_Changed(GetParent(hDlg), hDlg);
|
|
}
|
|
else
|
|
{
|
|
lpPropSheet->lParam &= ~AD_DefaultUser;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
default :
|
|
{
|
|
return (FALSE);
|
|
}
|
|
}
|
|
|
|
return (TRUE);
|
|
}
|