1433 lines
49 KiB
C++
1433 lines
49 KiB
C++
/**
|
|
|
|
Source File: Device Profile Management .CPP
|
|
|
|
Change History:
|
|
|
|
|
|
Implements the class which provides the various device profile management UI
|
|
|
|
Copyright (c) 1996 by Microsoft Corporation
|
|
|
|
A Pretty Penny Enterprises, Inc. Production
|
|
11-27-96 a-RobKj@microsoft.com coded it
|
|
|
|
**/
|
|
|
|
#include "ICMUI.H"
|
|
|
|
|
|
// This function is obtained from the April 1998 Knowledge Base
|
|
// Its purpose is to determine if the current user is an Administrator and therefore priveledged to change profile settings.
|
|
|
|
// BOOL IsAdmin(void)
|
|
|
|
// returns TRUE if user is an admin
|
|
// FALSE if user is not an admin
|
|
#if defined(_WIN95_)
|
|
// Always administrator on Windows 9x platform.
|
|
BOOL IsAdmin(void) {
|
|
return (TRUE);
|
|
}
|
|
#else
|
|
BOOL IsAdmin(void)
|
|
{
|
|
BOOL fReturn = FALSE;
|
|
PSID psidAdmin;
|
|
|
|
SID_IDENTIFIER_AUTHORITY SystemSidAuthority= SECURITY_NT_AUTHORITY;
|
|
|
|
if ( AllocateAndInitializeSid ( &SystemSidAuthority, 2, SECURITY_BUILTIN_DOMAIN_RID, DOMAIN_ALIAS_RID_ADMINS, 0, 0, 0, 0, 0, 0, &psidAdmin) )
|
|
{
|
|
if(!CheckTokenMembership( NULL, psidAdmin, &fReturn )) {
|
|
// explicitly disallow Admin Access if CheckTokenMembership fails.
|
|
fReturn = FALSE;
|
|
}
|
|
FreeSid ( psidAdmin);
|
|
}
|
|
|
|
return ( fReturn );
|
|
}
|
|
#endif // _WIN95_
|
|
|
|
|
|
/**
|
|
|
|
List Managment functions
|
|
|
|
The method used here is simlar to that used in the profile Managment sheets for managing device associations.
|
|
|
|
We (not royal, I mean the OS and I) manage two "to-do" lists and use these to show the user the anticipated result of these lists being applied.
|
|
The first list is a list of existing associations that are to be broken.
|
|
The second is one of new associations to be made.
|
|
This is coupled with a list of the current associations.
|
|
|
|
The "Removals" list is the indices of existing associations which will be borken.
|
|
The "Adds" list is a list of new profiles to be associated.
|
|
The "Profiles" list is the list of existing associations.
|
|
|
|
Adding and removing profiles could mean removing an item from one of the work lists (undoing a previous selection), or adding one.
|
|
Each time such a change is made, the profile list box is emptied and refilled.
|
|
This lets us avoid mapping removals and additions more directly.
|
|
|
|
When changes are commited, either with Apply or OK, we make or break associations as specified, then empty all of the lists, and rebuild the list of current associations.
|
|
|
|
We use the ITEMDATA of the UI list box to handle the associations.
|
|
This lets the list remain sorted.
|
|
|
|
All of the list management functions can be overriden, if needed.
|
|
**/
|
|
void CDeviceProfileManagement::InitList()
|
|
{
|
|
// Make sure the lists are empty.
|
|
m_cuaRemovals.Empty();
|
|
m_cpaAdds.Empty();
|
|
|
|
// Determine the associations for the target device.
|
|
ENUMTYPE et = {sizeof et, ENUM_TYPE_VERSION, ET_DEVICENAME, m_csDevice};
|
|
|
|
CProfile::Enumerate(et, m_cpaProfile);
|
|
}
|
|
|
|
// Fill the UI list of profiles
|
|
|
|
void CDeviceProfileManagement::FillList(DWORD dwFlags)
|
|
{
|
|
// Before reset list box, get current selection to restore later.
|
|
|
|
CString csSelect;
|
|
|
|
csSelect.Empty();
|
|
|
|
LRESULT idSelect = LB_ERR;
|
|
|
|
if ( !(dwFlags & DEVLIST_NOSELECT)) {
|
|
// Get current selected position.
|
|
|
|
idSelect = SendMessage(m_hwndList, LB_GETCURSEL, 0, 0);
|
|
|
|
// Get text length where currently selected, than allocate buffer for that.
|
|
|
|
DWORD dwLen = (DWORD) SendMessage(m_hwndList, LB_GETTEXTLEN, idSelect, 0);
|
|
TCHAR *pszSelect = new TCHAR[dwLen + 1];
|
|
|
|
// Get text itself.
|
|
|
|
if (pszSelect != NULL) {
|
|
if (SendMessage(m_hwndList, LB_GETTEXT, idSelect, (LPARAM) pszSelect) != LB_ERR) {
|
|
csSelect = pszSelect;
|
|
}
|
|
|
|
delete pszSelect;
|
|
}
|
|
}
|
|
|
|
// reset list box
|
|
|
|
SendMessage(m_hwndList, LB_RESETCONTENT, 0, 0);
|
|
|
|
// Fill the profile list box from the list of profiles
|
|
|
|
for (unsigned u = 0; u < m_cpaProfile.Count(); u++) {
|
|
// Don't list profiles tentatively disassociated...
|
|
|
|
for (unsigned uOut = 0; uOut < m_cuaRemovals.Count(); uOut++)
|
|
if (m_cuaRemovals[uOut] == u)
|
|
break;
|
|
|
|
if (uOut < m_cuaRemovals.Count())
|
|
continue; // Don't add this to list, it's been zapped!
|
|
|
|
LRESULT id = SendMessage(m_hwndList, LB_ADDSTRING, 0,
|
|
(LPARAM) (LPCTSTR) m_cpaProfile[u] -> GetName());
|
|
|
|
SendMessage(m_hwndList, LB_SETITEMDATA, id, u);
|
|
}
|
|
|
|
// Add the profiles that have been tentatively added...
|
|
|
|
for (u = 0; u < m_cpaAdds.Count(); u ++) {
|
|
LRESULT id = SendMessage(m_hwndList, LB_ADDSTRING, 0,
|
|
(LPARAM) (LPCTSTR) m_cpaAdds[u] -> GetName());
|
|
SendMessage(m_hwndList, LB_SETITEMDATA, id, u + m_cpaProfile.Count());
|
|
}
|
|
|
|
// If we have any profiles, select the first one
|
|
// Otherwise, disable the "Remove" button, as there's nothing to remove
|
|
|
|
unsigned itemCount = (m_cpaProfile.Count() + m_cpaAdds.Count() - m_cuaRemovals.Count());
|
|
if (itemCount) {
|
|
// The Remove button must remain disabled
|
|
// unless the user is Administrator.
|
|
// This code is specific to the Monitor Profile
|
|
// Property sheet.
|
|
|
|
if (!m_bReadOnly) {
|
|
EnableWindow(GetDlgItem(m_hwnd, RemoveButton), TRUE);
|
|
}
|
|
|
|
if ( !(dwFlags & DEVLIST_NOSELECT)) {
|
|
// Find out the string selected previously.
|
|
|
|
idSelect = LB_ERR;
|
|
|
|
if (!csSelect.IsEmpty()) {
|
|
idSelect = SendMessage(m_hwndList, LB_FINDSTRINGEXACT, (WPARAM) -1, (LPARAM) (LPCTSTR) csSelect);
|
|
}
|
|
|
|
// if could not find, just select first item.
|
|
if (idSelect == LB_ERR) {
|
|
idSelect = 0;
|
|
}
|
|
|
|
// Select it.
|
|
SendMessage(m_hwndList, LB_SETCURSEL, idSelect, 0);
|
|
}
|
|
} else {
|
|
HWND hwndRemove = GetDlgItem(m_hwnd, RemoveButton);
|
|
|
|
// If focus is on Remove, move it to Add button.
|
|
|
|
if (GetFocus() == hwndRemove) {
|
|
HWND hwndAdd = GetDlgItem(m_hwnd, AddButton);
|
|
|
|
SetFocus(hwndAdd);
|
|
SendMessage(hwndRemove, BM_SETSTYLE, BS_PUSHBUTTON, MAKELPARAM(TRUE, 0));
|
|
SendMessage(hwndAdd, BM_SETSTYLE, BS_DEFPUSHBUTTON, MAKELPARAM(TRUE, 0));
|
|
}
|
|
|
|
EnableWindow(hwndRemove, FALSE);
|
|
}
|
|
|
|
// Apply button needs to remain disabled unless the
|
|
// user has permision to make changes - ie. user
|
|
// is Administrator.
|
|
if ((dwFlags & DEVLIST_CHANGED) && !(m_bReadOnly)) {
|
|
EnableApplyButton();
|
|
SettingChanged(TRUE);
|
|
}
|
|
}
|
|
|
|
|
|
void CDeviceProfileManagement::GetDeviceTypeString(DWORD dwType,CString& csDeviceName)
|
|
{
|
|
DWORD id;
|
|
|
|
switch (dwType) {
|
|
case CLASS_MONITOR :
|
|
id = ClassMonitorString;
|
|
break;
|
|
case CLASS_PRINTER :
|
|
id = ClassPrinterString;
|
|
break;
|
|
case CLASS_SCANNER :
|
|
id = ClassScannerString;
|
|
break;
|
|
case CLASS_LINK :
|
|
id = ClassLinkString;
|
|
break;
|
|
case CLASS_ABSTRACT :
|
|
id = ClassAbstractString;
|
|
break;
|
|
case CLASS_NAMED :
|
|
id = ClassNamedString;
|
|
break;
|
|
case CLASS_COLORSPACE :
|
|
default :
|
|
id = ClassColorSpaceString;
|
|
break;
|
|
}
|
|
|
|
// Load string.
|
|
csDeviceName.Load(id);
|
|
}
|
|
|
|
// Constructor
|
|
|
|
CDeviceProfileManagement::CDeviceProfileManagement(LPCTSTR lpstrDevice, HINSTANCE hiWhere, int idPage, DWORD dwType)
|
|
{
|
|
m_csDevice = lpstrDevice;
|
|
m_dwType = dwType;
|
|
m_psp.hInstance = hiWhere;
|
|
m_psp.pszTemplate = MAKEINTRESOURCE(idPage);
|
|
|
|
// Setting m_bReadOnly to false enables functionality
|
|
|
|
m_bReadOnly = FALSE; // default setting is false
|
|
|
|
#if defined(_WIN95_)
|
|
// There is no way to detect printer supports CMYK or not on Win 9x.
|
|
m_bCMYK = TRUE;
|
|
#else
|
|
// we need to check the device capabilities
|
|
// and determine if we're trying to associate
|
|
// a cmyk printer profile to a printer that
|
|
// doesn't support it.
|
|
|
|
m_bCMYK = FALSE; // default setting - don't support cmyk
|
|
|
|
// if the device is a printer
|
|
|
|
if (m_dwType == CLASS_PRINTER) {
|
|
HDC hdcThis = CGlobals::GetPrinterHDC(m_csDevice);
|
|
|
|
// if the printer supports CMYK
|
|
if (hdcThis) {
|
|
if (GetDeviceCaps(hdcThis, COLORMGMTCAPS) & CM_CMYK_COLOR) {
|
|
m_bCMYK = TRUE;
|
|
}
|
|
DeleteDC(hdcThis);
|
|
}
|
|
}
|
|
|
|
#endif // defined(_WIN95_)
|
|
}
|
|
|
|
// UI initialization
|
|
|
|
BOOL CDeviceProfileManagement::OnInit()
|
|
{
|
|
InitList();
|
|
|
|
m_hwndList = GetDlgItem(m_hwnd, ProfileListControl);
|
|
|
|
FillList(DEVLIST_ONINIT);// Fill the profile list box
|
|
DisableApplyButton();// Disable apply button as default.
|
|
SettingChanged(FALSE);// Nothing changed, yet.
|
|
return TRUE;
|
|
}
|
|
|
|
// Command processing
|
|
|
|
BOOL CDeviceProfileManagement::OnCommand(WORD wNotifyCode, WORD wid, HWND hwndCtl)
|
|
{
|
|
switch (wNotifyCode) {
|
|
case BN_CLICKED:
|
|
switch (wid) {
|
|
case AddButton: {
|
|
unsigned i = 0, u = 0;
|
|
|
|
// Time to do the old OpenFile dialog stuff...
|
|
|
|
CAddProfileDialog capd(m_hwnd, m_psp.hInstance);
|
|
|
|
// See if a profile was selected
|
|
|
|
while(i < capd.ProfileCount()) {
|
|
// Check profile validity and device type
|
|
|
|
CProfile cpTemp(capd.ProfileName(i));
|
|
|
|
// CLASS_COLORSPACE and CLASS_MONITOR can be associated to
|
|
// any device. Other (CLASS_SCANNER, CLASS_PRINTER) only
|
|
// can be associated to much device.
|
|
|
|
if ( !cpTemp.IsValid() // Wrong profile type or invalid?
|
|
|| ( cpTemp.GetType() != m_dwType && cpTemp.GetType() != CLASS_COLORSPACE
|
|
#if 1 // ALLOW_MONITOR_PROFILE_TO_ANY_DEVICE
|
|
&& cpTemp.GetType() != CLASS_MONITOR
|
|
#endif
|
|
)
|
|
) {
|
|
// Throw up a message box to inform the user of this
|
|
|
|
if (cpTemp.IsValid())
|
|
{
|
|
CString csDeviceType; GetDeviceTypeString(m_dwType,csDeviceType);
|
|
CString csProfileType; GetDeviceTypeString(cpTemp.GetType(),csProfileType);
|
|
|
|
CGlobals::ReportEx(MismatchDeviceType, m_hwnd, FALSE,
|
|
MB_OK|MB_ICONEXCLAMATION, 3,
|
|
(LPTSTR)capd.ProfileNameAndExtension(i),
|
|
(LPTSTR)csProfileType,
|
|
(LPTSTR)csDeviceType);
|
|
}
|
|
else
|
|
{
|
|
CGlobals::ReportEx(InstFailedWithName, m_hwnd, FALSE, MB_OK|MB_ICONEXCLAMATION, 1, (LPTSTR)capd.ProfileNameAndExtension(i));
|
|
}
|
|
|
|
goto SkipToNext;
|
|
}
|
|
|
|
// See if the profile has already been listed for addition
|
|
|
|
for (u = 0; u < m_cpaAdds.Count(); u++) {
|
|
if (!lstrcmpi(m_cpaAdds[u] -> GetName(), cpTemp.GetName())) {
|
|
goto SkipToNext; // This profile is already added
|
|
}
|
|
}
|
|
|
|
// If this profile is on the existing list, either ignore
|
|
// or zap it from the removal list, as the case may be
|
|
|
|
for (u = 0; u < m_cpaProfile.Count(); u++) {
|
|
if (!lstrcmpi(m_cpaProfile[u] -> GetName(),
|
|
cpTemp.GetName())) {
|
|
// Is this one on the removal list?
|
|
for (unsigned uOut = 0;
|
|
uOut < m_cuaRemovals.Count();
|
|
uOut++) {
|
|
if (m_cuaRemovals[uOut] == u) {
|
|
// Was to be removed- undo that...
|
|
m_cuaRemovals.Remove(uOut);
|
|
FillList(DEVLIST_CHANGED);
|
|
break;
|
|
}
|
|
}
|
|
goto SkipToNext;
|
|
} // End of name in existing list
|
|
}
|
|
|
|
// We need to check the device capabilities
|
|
// and determine if we're trying to associate
|
|
// a cmyk printer profile to a printer that
|
|
// doesn't support it.
|
|
|
|
if ((!m_bCMYK) && (cpTemp.GetColorSpace() == SPACE_CMYK)) {
|
|
CGlobals::ReportEx(UnsupportedProfile, m_hwnd, FALSE, MB_OK|MB_ICONEXCLAMATION, 2, (LPTSTR)m_csDevice, (LPTSTR)capd.ProfileNameAndExtension(i));
|
|
goto SkipToNext;
|
|
}
|
|
|
|
// Add this profile to the list, item (max orig + index)
|
|
m_cpaAdds.Add(capd.ProfileName(i));
|
|
|
|
// Change has been made, update the list
|
|
FillList(DEVLIST_CHANGED);
|
|
SkipToNext:
|
|
i++;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
case RemoveButton: {
|
|
// Remove the selected profile
|
|
|
|
LRESULT id = SendMessage(m_hwndList, LB_GETCURSEL, 0, 0);
|
|
unsigned u = (unsigned) SendMessage(m_hwndList, LB_GETITEMDATA, id, 0);
|
|
|
|
// If this is a tentative add, just drop it, otherwise
|
|
// note that it's been removed...
|
|
|
|
if (u >= m_cpaProfile.Count())
|
|
m_cpaAdds.Remove(u - m_cpaProfile.Count());
|
|
else
|
|
m_cuaRemovals.Add(u);
|
|
|
|
// That's it- just update the display, now...
|
|
|
|
FillList(DEVLIST_CHANGED);
|
|
|
|
// explicitly set the position of the current selection
|
|
// after the list has been recomputed.
|
|
|
|
int listsize = m_cpaProfile.Count()+m_cpaAdds.Count()-m_cuaRemovals.Count();
|
|
if (id >= listsize) id = listsize-1;
|
|
if (id < 0) id = 0;
|
|
SendMessage(m_hwndList, LB_SETCURSEL, id, 0);
|
|
|
|
return TRUE;
|
|
}
|
|
}
|
|
break;
|
|
case LBN_SELCHANGE: {
|
|
LRESULT id = SendMessage(m_hwndList, LB_GETCURSEL, 0, 0);
|
|
if (id == -1) {
|
|
EnableWindow(GetDlgItem(m_hwnd, RemoveButton), FALSE);
|
|
} else {
|
|
|
|
// The Remove button must remain disabled on a monitor
|
|
// profile property page if the user isn't the
|
|
// Administrator, otherwise enable remove button.
|
|
|
|
EnableWindow(GetDlgItem(m_hwnd, RemoveButton), !m_bReadOnly);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
// Property Sheet notification processing
|
|
|
|
BOOL CDeviceProfileManagement::OnNotify(int idCtrl, LPNMHDR pnmh) {
|
|
switch (pnmh -> code) {
|
|
case PSN_APPLY:
|
|
DisableApplyButton();
|
|
|
|
if (SettingChanged()) {
|
|
// Apply the changes the user has made...
|
|
|
|
SettingChanged(FALSE);
|
|
|
|
while (m_cpaAdds.Count()) {
|
|
if (!m_cpaAdds[0] -> IsInstalled()) {
|
|
m_cpaAdds[0] -> Install();
|
|
}
|
|
m_cpaAdds[0] -> Associate(m_csDevice);
|
|
m_cpaAdds.Remove(0);
|
|
}
|
|
|
|
// Now do the removals (actually just dissociations)
|
|
|
|
while (m_cuaRemovals.Count()) {
|
|
m_cpaProfile[m_cuaRemovals[0]] -> Dissociate(m_csDevice);
|
|
m_cuaRemovals.Remove(0);
|
|
}
|
|
|
|
InitList();
|
|
FillList();
|
|
|
|
SetWindowLongPtr(m_hwnd, DWLP_MSGRESULT, PSNRET_NOERROR);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
// This hook procedure both forces the use of the old-style common dialog
|
|
// and changes the OK button to an Add button. The actual button text is
|
|
// a string resource, and hence localizable.
|
|
|
|
UINT_PTR APIENTRY CAddProfileDialog::OpenFileHookProc(HWND hDlg, UINT uMessage, WPARAM wp, LPARAM lp)
|
|
{
|
|
switch (uMessage) {
|
|
case WM_INITDIALOG: {
|
|
CString csAddButton;
|
|
OPENFILENAME *pofn = (OPENFILENAME *) lp;
|
|
|
|
csAddButton.Load(AddButtonText);
|
|
SetDlgItemText(GetParent(hDlg), IDOK, csAddButton);
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
// Once again, a constructor that actually does most of the work!
|
|
|
|
TCHAR gacColorDir[MAX_PATH] = _TEXT("\0");
|
|
TCHAR gacFilter[MAX_PATH] = _TEXT("\0");
|
|
|
|
CAddProfileDialog::CAddProfileDialog(HWND hwndOwner, HINSTANCE hi)
|
|
{
|
|
TCHAR tempBuffer[MAX_PATH*10];
|
|
|
|
// Empty the profile list.
|
|
|
|
csa_Files.Empty();
|
|
|
|
// Prepare file filter (if not yet).
|
|
|
|
if (gacFilter[0] == NULL) {
|
|
|
|
ULONG offset; /* 32bits is enough even for sundown */
|
|
CString csIccFilter; CString csAllFilter;
|
|
|
|
// If the filter is not built yet, build it here.
|
|
|
|
csIccFilter.Load(IccProfileFilterString);
|
|
csAllFilter.Load(AllProfileFilterString);
|
|
offset = 0;
|
|
lstrcpy(gacFilter+offset, csIccFilter);
|
|
offset += lstrlen(csIccFilter)+1;
|
|
lstrcpy(gacFilter+offset, TEXT("*.icm;*.icc"));
|
|
offset += lstrlen(TEXT("*.icm;*.icc"))+1;
|
|
lstrcpy(gacFilter+offset, csAllFilter);
|
|
offset += lstrlen(csAllFilter)+1;
|
|
lstrcpy(gacFilter+offset, TEXT("*.*"));
|
|
offset += lstrlen(TEXT("*.*"))+1;
|
|
*(gacFilter+offset) = TEXT('\0');
|
|
}
|
|
|
|
if (gacColorDir[0] == _TEXT('\0')) {
|
|
DWORD dwcbDir = MAX_PATH;
|
|
GetColorDirectory(NULL, gacColorDir, &dwcbDir);
|
|
}
|
|
|
|
// Time to do the old OpenFile dialog stuff...
|
|
CString csTitle; csTitle.Load(AddProfileAssociation);
|
|
|
|
// Set initial filename as null.
|
|
memset(tempBuffer, 0, sizeof tempBuffer);
|
|
|
|
OPENFILENAME ofn = {
|
|
sizeof ofn, hwndOwner, hi,
|
|
gacFilter,
|
|
NULL, 0, 1,
|
|
tempBuffer, sizeof tempBuffer / sizeof tempBuffer[0],
|
|
NULL, 0,
|
|
gacColorDir,
|
|
csTitle,
|
|
OFN_ALLOWMULTISELECT | OFN_EXPLORER | OFN_HIDEREADONLY |
|
|
OFN_FILEMUSTEXIST | OFN_PATHMUSTEXIST | OFN_ENABLEHOOK,
|
|
0, 0,
|
|
_TEXT("icm"),
|
|
(LPARAM) this, OpenFileHookProc, NULL};
|
|
|
|
if (!GetOpenFileName(&ofn)) {
|
|
if (CommDlgExtendedError() == FNERR_BUFFERTOOSMALL) {
|
|
CGlobals::Report(TooManyFileSelected);
|
|
}
|
|
} else {
|
|
if (tempBuffer[0] != TEXT('\0')) {
|
|
|
|
TCHAR *pPath = tempBuffer;
|
|
TCHAR *pFile = tempBuffer + lstrlen(tempBuffer) + 1;
|
|
|
|
// remember the last access-ed directory.
|
|
|
|
memset(gacColorDir, 0, sizeof pPath);
|
|
memcpy(gacColorDir, pPath, ofn.nFileOffset*sizeof(TCHAR));
|
|
|
|
if (*pFile) {
|
|
TCHAR workBuffer[MAX_PATH];
|
|
|
|
// This is multiple-selection
|
|
// Work through the buufer to build profile file list.
|
|
|
|
while (*pFile) {
|
|
lstrcpy(workBuffer,pPath);
|
|
lstrcat(workBuffer,TEXT("\\"));
|
|
lstrcat(workBuffer,pFile);
|
|
|
|
AddProfile(workBuffer);// Insert built profile pathname
|
|
pFile = pFile + lstrlen(pFile) + 1;// Move on to next.
|
|
}
|
|
}
|
|
else {
|
|
// Single selection case.
|
|
AddProfile(pPath);
|
|
|
|
#if HIDEYUKN_DBG
|
|
MessageBox(NULL,pPath,TEXT(""),MB_OK);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
// Printer Profile Management
|
|
|
|
CONST DWORD PrinterUIHelpIds[] = {
|
|
AddButton, IDH_PRINTERUI_ADD,
|
|
RemoveButton, IDH_PRINTERUI_REMOVE,
|
|
ProfileListControl, IDH_PRINTERUI_LIST,
|
|
|
|
#if !defined(_WIN95_)
|
|
ProfileListControlText, IDH_PRINTERUI_LIST,
|
|
PrinterUIIcon, IDH_DISABLED,
|
|
DescriptionText, IDH_DISABLED,
|
|
DefaultButton, IDH_PRINTERUI_DEFAULTBTN,
|
|
AutoSelButton, IDH_PRINTERUI_AUTOMATIC,
|
|
AutoSelText, IDH_PRINTERUI_AUTOMATIC,
|
|
ManualSelButton, IDH_PRINTERUI_MANUAL,
|
|
ManualSelText, IDH_PRINTERUI_MANUAL,
|
|
DefaultProfileText, IDH_PRINTERUI_DEFAULTTEXT,
|
|
DefaultProfile, IDH_PRINTERUI_DEFAULTTEXT,
|
|
#endif
|
|
0, 0
|
|
};
|
|
|
|
// Initialize lists override- call the base class, then set the default
|
|
|
|
void CPrinterProfileManagement::InitList() {
|
|
CDeviceProfileManagement::InitList();
|
|
m_uDefault = m_cpaProfile.Count() ? 0 : (unsigned) -1;
|
|
}
|
|
|
|
// Fill list override- write the correct default and call the base function
|
|
|
|
void CPrinterProfileManagement::FillList(DWORD dwFlags)
|
|
{
|
|
// If we are initializing list box, we want to put focus on
|
|
// "default" profile. here we won't below FillList set focus to first one.
|
|
if (dwFlags & DEVLIST_ONINIT) {
|
|
dwFlags |= DEVLIST_NOSELECT;
|
|
}
|
|
|
|
CDeviceProfileManagement::FillList(dwFlags);
|
|
|
|
// There is either no default profile, an existing profile is the
|
|
// default, or a newly selected one is. Some people just like the
|
|
// selection operator.
|
|
|
|
// if there is only 1 profile in list box, we treat it as default profile.
|
|
|
|
if (SendMessage(m_hwndList,LB_GETCOUNT,0,0) == 1) {
|
|
m_uDefault = (unsigned)SendMessage(m_hwndList, LB_GETITEMDATA, 0, 0);
|
|
}
|
|
|
|
if (m_uDefault == -1) {
|
|
// There is no profile associated for this device.
|
|
|
|
CString csNoProfile;
|
|
csNoProfile.Load(NoProfileString);
|
|
SetDlgItemText(m_hwnd, DefaultProfile, csNoProfile);
|
|
} else {
|
|
// If the default has been deleted, set default as last in list.
|
|
|
|
if (m_uDefault >= (m_cpaProfile.Count() + m_cpaAdds.Count())) {
|
|
m_uDefault = (m_cpaProfile.Count() + m_cpaAdds.Count()) - 1;
|
|
}
|
|
|
|
// Put default profile name in UI.
|
|
|
|
CProfile *pcpDefault = (m_uDefault < m_cpaProfile.Count()) ? \
|
|
m_cpaProfile[m_uDefault] : \
|
|
m_cpaAdds[m_uDefault - m_cpaProfile.Count()];
|
|
|
|
SetDlgItemText(m_hwnd, DefaultProfile, pcpDefault -> GetName());
|
|
|
|
LRESULT idSelect = SendMessage(m_hwndList, LB_FINDSTRINGEXACT, (WPARAM) -1, (LPARAM) (LPCTSTR) pcpDefault -> GetName());
|
|
|
|
// if could not find, just select first item.
|
|
|
|
if (idSelect == LB_ERR) {
|
|
idSelect = 0;
|
|
}
|
|
|
|
// Select it.
|
|
|
|
SendMessage(m_hwndList, LB_SETCURSEL, idSelect, 0);
|
|
}
|
|
|
|
// 03-08-1997 Bob_Kjelgaard@Prodigy.Net Memphis RAID 18420
|
|
// Disable the Default button if there aren't any profiles
|
|
|
|
if (m_bManualMode && m_bAdminAccess) {
|
|
EnableWindow(GetDlgItem(m_hwnd, DefaultButton), m_cpaAdds.Count() + m_cpaProfile.Count() - m_cuaRemovals.Count());
|
|
}
|
|
}
|
|
|
|
// Printer Profile Management class constructor- doesn't need any individual
|
|
// code at the moment.
|
|
|
|
CPrinterProfileManagement::CPrinterProfileManagement(LPCTSTR lpstrName, HINSTANCE hiWhere) :
|
|
CDeviceProfileManagement(lpstrName, hiWhere, PrinterUI, CLASS_PRINTER) {
|
|
}
|
|
|
|
// This class overrides OnInit so it can disable the UI if the user lacks
|
|
// authority to make changes.
|
|
|
|
BOOL CPrinterProfileManagement::OnInit()
|
|
{
|
|
// Call the base class routine first, as it does most of the work...
|
|
|
|
CDeviceProfileManagement::OnInit();
|
|
|
|
DWORD dwSize = sizeof(DWORD);
|
|
|
|
// Query current mode.
|
|
|
|
if (!InternalGetDeviceConfig((LPCTSTR)m_csDevice, CLASS_PRINTER, MSCMS_PROFILE_ENUM_MODE, &m_bManualMode, &dwSize)) {
|
|
// Auto selection mode as default.
|
|
m_bManualMode = FALSE;
|
|
}
|
|
|
|
// Now, see if we have sufficient authority to administer the printer
|
|
|
|
HANDLE hPrinter;
|
|
PRINTER_DEFAULTS pd = {NULL, NULL, PRINTER_ACCESS_ADMINISTER};
|
|
|
|
m_bAdminAccess = TRUE;
|
|
m_bLocalPrinter = TRUE;
|
|
|
|
if (OpenPrinter(const_cast<LPTSTR> ((LPCTSTR) m_csDevice), &hPrinter, &pd)) {
|
|
|
|
// We can administer the printer- proceed in the normal way.
|
|
|
|
#if !defined(_WIN95_)
|
|
|
|
// If the printer is "Network Printer", we don't allow user to install
|
|
// or uninstall color profile.
|
|
|
|
BYTE StackPrinterData[sizeof(PRINTER_INFO_4)+MAX_PATH*2];
|
|
PBYTE pPrinterData = StackPrinterData;
|
|
BOOL bSuccess = TRUE;
|
|
DWORD dwReturned;
|
|
|
|
if (!GetPrinter(hPrinter, 4, pPrinterData, sizeof(StackPrinterData), &dwReturned)) {
|
|
if ((GetLastError() == ERROR_INSUFFICIENT_BUFFER) && (pPrinterData = (PBYTE) LocalAlloc(LPTR, dwReturned))) {
|
|
if (GetPrinter(hPrinter, 4, pPrinterData, dwReturned, &dwReturned)) {
|
|
bSuccess = TRUE;
|
|
}
|
|
}
|
|
} else {
|
|
bSuccess = TRUE;
|
|
}
|
|
|
|
if (bSuccess)
|
|
{
|
|
m_bLocalPrinter = ((PRINTER_INFO_4 *)pPrinterData)->pServerName ? FALSE : TRUE;
|
|
}
|
|
else
|
|
{
|
|
m_bAdminAccess = FALSE;
|
|
}
|
|
|
|
if (pPrinterData && (pPrinterData != StackPrinterData))
|
|
{
|
|
LocalFree(pPrinterData);
|
|
}
|
|
|
|
#endif // !defined(_WIN95_)
|
|
|
|
ClosePrinter(hPrinter);
|
|
} else {
|
|
m_bAdminAccess = FALSE;
|
|
}
|
|
|
|
// How many profile in listbox ?
|
|
|
|
LRESULT itemCount = SendMessage(m_hwndList, LB_GETCOUNT, 0, 0);
|
|
if (itemCount == LB_ERR) itemCount = 0;
|
|
|
|
// make sure the ancestor list code behaves correctly.
|
|
// You need Admin Access and a Local Printer to be able to add/remove profiles
|
|
|
|
m_bReadOnly = !(m_bAdminAccess && m_bLocalPrinter);
|
|
|
|
// Enable/Disable the controls (if needed)
|
|
|
|
CheckDlgButton(m_hwnd, AutoSelButton, m_bManualMode ? BST_UNCHECKED : BST_CHECKED);
|
|
CheckDlgButton(m_hwnd, ManualSelButton, m_bManualMode ? BST_CHECKED : BST_UNCHECKED);
|
|
|
|
// Only administrator can change 'auto','manual' configuration.
|
|
|
|
EnableWindow(GetDlgItem(m_hwnd, AutoSelButton), m_bAdminAccess && m_bLocalPrinter);
|
|
EnableWindow(GetDlgItem(m_hwnd, ManualSelButton), m_bAdminAccess && m_bLocalPrinter);
|
|
|
|
// Only administrator and printer is at local, can install/uninstall color profile.
|
|
|
|
EnableWindow(GetDlgItem(m_hwnd, AddButton), m_bAdminAccess && m_bLocalPrinter);
|
|
EnableWindow(GetDlgItem(m_hwnd, RemoveButton), m_bAdminAccess && m_bLocalPrinter && itemCount);
|
|
|
|
EnableWindow(m_hwndList, m_bAdminAccess);
|
|
EnableWindow(GetDlgItem(m_hwnd, DefaultProfileText), m_bAdminAccess);
|
|
EnableWindow(GetDlgItem(m_hwnd, DefaultProfile), m_bAdminAccess);
|
|
|
|
// Only with manual mode, these controls are enabled.
|
|
|
|
EnableWindow(GetDlgItem(m_hwnd, DefaultButton), m_bAdminAccess && m_bManualMode && m_bLocalPrinter && itemCount);
|
|
|
|
if (!m_bAdminAccess) {
|
|
// Set the focus to the OK button
|
|
|
|
SetFocus(GetDlgItem(m_hwndSheet, IDOK));
|
|
return FALSE; // Because we moved the focus!
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
// Command processing- we never let them click into the edit control, to
|
|
// prevent them from editing it.
|
|
|
|
BOOL CPrinterProfileManagement::OnCommand(WORD wNotifyCode, WORD wid, HWND hwndCtl)
|
|
{
|
|
switch (wNotifyCode) {
|
|
case LBN_DBLCLK: {
|
|
// Retrieve the ID of the new default profile
|
|
// only accept dblclk changes if the dialog
|
|
// is not read only - i.e. user is admin
|
|
|
|
if (m_bManualMode) {
|
|
int id = (int)SendMessage(m_hwndList, LB_GETCURSEL, 0, 0);
|
|
m_uDefault = (unsigned)SendMessage(m_hwndList, LB_GETITEMDATA, id, 0);
|
|
|
|
// Change has been made, update the list
|
|
|
|
FillList(DEVLIST_CHANGED);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
case BN_CLICKED:
|
|
switch (wid) {
|
|
case AutoSelButton:
|
|
case ManualSelButton: {
|
|
|
|
// How many profile in listbox ?
|
|
|
|
LRESULT itemCount = SendMessage(m_hwndList, LB_GETCOUNT, 0, 0);
|
|
if (itemCount == LB_ERR) itemCount = 0;
|
|
|
|
m_bManualMode = (wid == ManualSelButton) ? TRUE : FALSE;
|
|
|
|
// Only with manual mode, these controls are enabled.
|
|
|
|
EnableWindow(GetDlgItem(m_hwnd, DefaultButton), m_bManualMode && itemCount);
|
|
|
|
// Configuarion has been changed, enable apply button.
|
|
|
|
EnableApplyButton();
|
|
SettingChanged(TRUE);
|
|
|
|
return TRUE;
|
|
}
|
|
case RemoveButton: {
|
|
// Make sure we've tracked the default profile correctly
|
|
// when a profile is removed.
|
|
// All cases break, because we then want the base class to
|
|
// process this message.
|
|
|
|
LRESULT id = SendMessage(m_hwndList, LB_GETCURSEL, 0, 0);
|
|
|
|
unsigned uTarget = (unsigned)SendMessage(m_hwndList, LB_GETITEMDATA, id, 0);
|
|
|
|
if (uTarget > m_uDefault || m_uDefault == (unsigned) -1)
|
|
break; // Nothing here to worry about
|
|
|
|
if (m_uDefault == uTarget) {
|
|
|
|
if (CGlobals::ReportEx(AskRemoveDefault, m_hwnd, FALSE, MB_YESNO|MB_ICONEXCLAMATION,0) == IDYES) {
|
|
|
|
// The default has been deleted- the profile
|
|
// at the top of monitor profile list will be
|
|
// made the default profile, if we have
|
|
|
|
LRESULT itemCount = SendMessage(m_hwndList, LB_GETCOUNT, 0, 0);
|
|
|
|
if ((itemCount != LB_ERR) && (itemCount > 1)) {
|
|
m_uDefault = (unsigned)SendMessage(m_hwndList, LB_GETITEMDATA, 0, 0);
|
|
if (m_uDefault == uTarget) {
|
|
m_uDefault = (unsigned)SendMessage(m_hwndList, LB_GETITEMDATA, 1, 0);
|
|
}
|
|
} else {
|
|
m_uDefault = -1;
|
|
}
|
|
|
|
break;
|
|
} else {
|
|
return TRUE; // opration cancelled.
|
|
}
|
|
}
|
|
|
|
if (uTarget < m_cpaProfile.Count())
|
|
break; // We're fine
|
|
|
|
// Must be an added profile below us in the list was
|
|
// zapped- we need to decrement ourselves.
|
|
|
|
m_uDefault--;
|
|
break;
|
|
}
|
|
case DefaultButton: {
|
|
LRESULT id = SendMessage(m_hwndList, LB_GETCURSEL, 0, 0);
|
|
|
|
m_uDefault = (unsigned)SendMessage(m_hwndList, LB_GETITEMDATA, id, 0);
|
|
|
|
// Change has been made, update the list
|
|
FillList(DEVLIST_CHANGED);
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
// Deliberate fall-through (use a break if you add a case here)
|
|
}
|
|
|
|
// Use common command handling if not handled above
|
|
return CDeviceProfileManagement::OnCommand(wNotifyCode, wid, hwndCtl);
|
|
}
|
|
|
|
// Property Sheet notification processing
|
|
|
|
BOOL CPrinterProfileManagement::OnNotify(int idCtrl, LPNMHDR pnmh)
|
|
{
|
|
switch (pnmh -> code) {
|
|
case PSN_APPLY: {
|
|
DisableApplyButton();
|
|
|
|
// If nothing changed, nothing need to do.
|
|
|
|
if (!SettingChanged())
|
|
return TRUE;
|
|
|
|
if (m_bManualMode) {
|
|
|
|
// If the user hasn't selected a default, and we have
|
|
// associated profiles, then we can't allow this.
|
|
|
|
// 03-08-1997 A-RobKj Fix for Memphis RAID 18416- if there's
|
|
// only one profile left, then it must be the default.
|
|
|
|
if (m_uDefault == (unsigned) -1 && (m_cpaAdds.Count() + m_cpaProfile.Count() - m_cuaRemovals.Count()) > 1) {
|
|
CGlobals::Report(NoDefaultProfile, m_hwndSheet);
|
|
SetWindowLongPtr(m_hwnd, DWLP_MSGRESULT, PSNRET_INVALID_NOCHANGEPAGE);
|
|
break;
|
|
}
|
|
|
|
// !!! This behavior is hardly depend on EnumColorProfiles() API !!!
|
|
|
|
// OK, if the default profile has changed, we have to delete default
|
|
// profile associations, and do the association for default profile
|
|
// in "last".
|
|
|
|
// Let the base class handle the cases where the default hasn't
|
|
// changed, or we started with no profiles and still have none.
|
|
|
|
// 03-08-1997 Sleazy code note. The case where no default is
|
|
// selected but only one is assigned will now fall here. Since the
|
|
// default happens to be the "last", and there only is one, letting
|
|
// the base class handle it is not a problem. The list filling
|
|
// code will take care of the rest for us.
|
|
|
|
if (m_uDefault == (unsigned) -1) break;
|
|
|
|
// Remove default first (if default is associated), then associate later.
|
|
if (m_uDefault < m_cpaProfile.Count())
|
|
m_cpaProfile[m_uDefault] -> Dissociate(m_csDevice);
|
|
|
|
// Now do the other removals (actually just dissociations)
|
|
for (unsigned u = 0; u < m_cuaRemovals.Count(); u++) {
|
|
m_cpaProfile[m_cuaRemovals[u]] -> Dissociate(m_csDevice);
|
|
}
|
|
|
|
// Add in the new ones
|
|
for (u = 0; u < m_cpaAdds.Count(); u++) {
|
|
if (m_uDefault >= m_cpaProfile.Count())
|
|
if (u == (m_uDefault - m_cpaProfile.Count()))
|
|
continue; // this is default, will be done later
|
|
|
|
// OK, add it back in...
|
|
m_cpaAdds[u] -> Associate(m_csDevice);
|
|
}
|
|
|
|
// Finally, associate back default profile.
|
|
if (m_uDefault < m_cpaProfile.Count())
|
|
m_cpaProfile[m_uDefault] -> Associate(m_csDevice);
|
|
else
|
|
m_cpaAdds[m_uDefault - m_cpaProfile.Count()] -> Associate(m_csDevice);
|
|
|
|
// Update the various working structures...
|
|
InitList();
|
|
FillList();
|
|
|
|
SetWindowLongPtr(m_hwnd, DWLP_MSGRESULT, PSNRET_NOERROR);
|
|
|
|
// Now, we have updated settings.
|
|
SettingChanged(FALSE);
|
|
}
|
|
|
|
// Update "auto/manual" status.
|
|
InternalSetDeviceConfig((LPCTSTR)m_csDevice, CLASS_PRINTER, MSCMS_PROFILE_ENUM_MODE, &m_bManualMode, sizeof(DWORD));
|
|
}
|
|
}
|
|
|
|
// Let the base class handle everything else
|
|
return CDeviceProfileManagement::OnNotify(idCtrl, pnmh);
|
|
}
|
|
|
|
// Context-sensitive help handler
|
|
|
|
BOOL CPrinterProfileManagement::OnHelp(LPHELPINFO pHelp) {
|
|
if (pHelp->iContextType == HELPINFO_WINDOW) {
|
|
WinHelp((HWND) pHelp->hItemHandle, WINDOWS_HELP_FILE, HELP_WM_HELP, (ULONG_PTR) (LPSTR) PrinterUIHelpIds);
|
|
}
|
|
|
|
return (TRUE);
|
|
}
|
|
|
|
BOOL CPrinterProfileManagement::OnContextMenu(HWND hwnd) {
|
|
WinHelp(hwnd, WINDOWS_HELP_FILE, HELP_CONTEXTMENU, (ULONG_PTR) (LPSTR) PrinterUIHelpIds);
|
|
return (TRUE);
|
|
}
|
|
|
|
// Scanner Profile Management
|
|
// Scanner Profile Management class constructor- doesn't need any individual
|
|
// code at the moment.
|
|
|
|
// Scanner Profile Management
|
|
|
|
CONST DWORD ScannerUIHelpIds[] = {
|
|
#if !defined(_WIN95_)
|
|
AddButton, IDH_SCANNERUI_ADD,
|
|
RemoveButton, IDH_SCANNERUI_REMOVE,
|
|
ProfileListControl, IDH_SCANNERUI_LIST,
|
|
ProfileListControlText, IDH_SCANNERUI_LIST,
|
|
#endif
|
|
0, 0
|
|
};
|
|
|
|
CScannerProfileManagement::CScannerProfileManagement(LPCTSTR lpstrName, HINSTANCE hiWhere) :
|
|
CDeviceProfileManagement(lpstrName, hiWhere, ScannerUI, CLASS_SCANNER) {
|
|
m_bReadOnly = !IsAdmin();
|
|
}
|
|
|
|
// This class overrides OnInit so it can disable the UI if the user lacks
|
|
// authority to make changes.
|
|
|
|
BOOL CScannerProfileManagement::OnInit() {
|
|
|
|
// Call the base class routine first, as it does most of the work...
|
|
|
|
CDeviceProfileManagement::OnInit();
|
|
|
|
// Now, see if we have sufficient authority to administer the scanner
|
|
|
|
if (m_bReadOnly) {
|
|
// User is not Admin, Disable all of the controls
|
|
|
|
EnableWindow(GetDlgItem(m_hwnd, AddButton), FALSE);
|
|
EnableWindow(GetDlgItem(m_hwnd, RemoveButton), FALSE);
|
|
|
|
// Set the focus to the OK button
|
|
SetFocus(GetDlgItem(m_hwndSheet, IDOK));
|
|
return FALSE; // Because we moved the focus!
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
// Context-sensitive help handler
|
|
|
|
BOOL CScannerProfileManagement::OnHelp(LPHELPINFO pHelp) {
|
|
if (pHelp->iContextType == HELPINFO_WINDOW) {
|
|
WinHelp((HWND) pHelp->hItemHandle, WINDOWS_HELP_FILE, HELP_WM_HELP, (ULONG_PTR) (LPSTR) ScannerUIHelpIds);
|
|
}
|
|
|
|
return (TRUE);
|
|
}
|
|
|
|
BOOL CScannerProfileManagement::OnContextMenu(HWND hwnd) {
|
|
WinHelp(hwnd, WINDOWS_HELP_FILE, HELP_CONTEXTMENU, (ULONG_PTR) (LPSTR) ScannerUIHelpIds);
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
// Monitor Profile Management class- since the mechanism for default
|
|
// profile manipulation is a bit sleazy, so is some of this code.
|
|
|
|
CONST DWORD MonitorUIHelpIds[] = {
|
|
AddButton, IDH_MONITORUI_ADD,
|
|
RemoveButton, IDH_MONITORUI_REMOVE,
|
|
DefaultButton, IDH_MONITORUI_DEFAULT,
|
|
ProfileListControl, IDH_MONITORUI_LIST,
|
|
#if !defined(_WIN95_)
|
|
ProfileListControlText, IDH_MONITORUI_LIST,
|
|
MonitorName, IDH_MONITORUI_DISPLAY,
|
|
MonitorNameText, IDH_MONITORUI_DISPLAY,
|
|
DefaultProfile, IDH_MONITORUI_PROFILE,
|
|
DefaultProfileText, IDH_MONITORUI_PROFILE,
|
|
#endif
|
|
0, 0
|
|
};
|
|
|
|
// Initialize lists override- call the base class, then set the default
|
|
|
|
void CMonitorProfileManagement::InitList()
|
|
{
|
|
CDeviceProfileManagement::InitList();
|
|
m_uDefault = m_cpaProfile.Count() ? 0 : (unsigned) -1;
|
|
}
|
|
|
|
// Fill list override- write the correct default and call the base function
|
|
|
|
void CMonitorProfileManagement::FillList(DWORD dwFlags)
|
|
{
|
|
// If we are initializing list box, we want to put focus on
|
|
// "default" profile. here we won't below FillList set focus
|
|
// to first one.
|
|
|
|
if (dwFlags & DEVLIST_ONINIT) {
|
|
dwFlags |= DEVLIST_NOSELECT;
|
|
}
|
|
|
|
CDeviceProfileManagement::FillList(dwFlags);
|
|
|
|
// There is either no default profile, an existing profile is the
|
|
// default, or a newly selected one is. Some people just like the
|
|
// selection operator.
|
|
|
|
// if there is only 1 profile in list box, we treat it as default profile.
|
|
|
|
if (SendMessage(m_hwndList,LB_GETCOUNT,0,0) == 1) {
|
|
m_uDefault = (unsigned)SendMessage(m_hwndList, LB_GETITEMDATA, 0, 0);
|
|
}
|
|
|
|
if (m_uDefault == -1) {
|
|
|
|
// There is no profile associated for this device.
|
|
|
|
CString csNoProfile;
|
|
csNoProfile.Load(NoProfileString);
|
|
SetDlgItemText(m_hwnd, DefaultProfile, csNoProfile);
|
|
} else {
|
|
|
|
// If the default has been deleted, set default as last in list.
|
|
|
|
if (m_uDefault >= (m_cpaProfile.Count() + m_cpaAdds.Count())) {
|
|
m_uDefault = (m_cpaProfile.Count() + m_cpaAdds.Count()) - 1;
|
|
}
|
|
|
|
// Put default profile name in UI.
|
|
|
|
CProfile *pcpDefault = (m_uDefault < m_cpaProfile.Count()) ? \
|
|
m_cpaProfile[m_uDefault] : \
|
|
m_cpaAdds[m_uDefault - m_cpaProfile.Count()];
|
|
|
|
SetDlgItemText(m_hwnd, DefaultProfile, pcpDefault -> GetName());
|
|
|
|
// If we are initialing list box, put focus on default profile.
|
|
|
|
if (dwFlags & DEVLIST_ONINIT) {
|
|
LRESULT idSelect = SendMessage(m_hwndList, LB_FINDSTRINGEXACT, (WPARAM) -1, (LPARAM) (LPCTSTR) pcpDefault -> GetName());
|
|
|
|
// if could not find, just select first item.
|
|
|
|
if (idSelect == LB_ERR) {
|
|
idSelect = 0;
|
|
}
|
|
|
|
// Select it.
|
|
|
|
SendMessage(m_hwndList, LB_SETCURSEL, idSelect, 0);
|
|
}
|
|
}
|
|
|
|
// 03-08-1997 Bob_Kjelgaard@Prodigy.Net Memphis RAID 18420
|
|
// Disable the Default button if there aren't any profiles
|
|
|
|
// We do it here, because this gets called any time the list changes.
|
|
// only allow Default button to be enabled if
|
|
// the dialog isn't read only.
|
|
// the remove button should remain dissabled
|
|
// under all conditions while the user is not
|
|
// administrator.
|
|
|
|
if (m_bReadOnly) {
|
|
EnableWindow(GetDlgItem(m_hwnd, RemoveButton), FALSE);
|
|
} else {
|
|
EnableWindow(GetDlgItem(m_hwnd, DefaultButton), m_cpaAdds.Count() + m_cpaProfile.Count() - m_cuaRemovals.Count());
|
|
}
|
|
}
|
|
|
|
// Constructor
|
|
|
|
CMonitorProfileManagement::CMonitorProfileManagement(LPCTSTR lpstrName, LPCTSTR lpstrFriendlyName, HINSTANCE hiWhere) :
|
|
CDeviceProfileManagement(lpstrName, hiWhere, MonitorUI, CLASS_MONITOR) {
|
|
// if the user is not the administrator,
|
|
// make this property sheet read only.
|
|
|
|
m_bReadOnly = !IsAdmin();
|
|
|
|
// Keep friendly name in MonitorProfileManagement class.
|
|
|
|
m_csDeviceFriendlyName = lpstrFriendlyName;
|
|
}
|
|
|
|
// UI Initialization
|
|
|
|
BOOL CMonitorProfileManagement::OnInit()
|
|
{
|
|
// Do common initializations
|
|
CDeviceProfileManagement::OnInit();
|
|
|
|
// Mark the device name in the space provided
|
|
SetDlgItemText(m_hwnd, MonitorName, m_csDeviceFriendlyName);
|
|
|
|
// Now, see if we have sufficient authority to administer the monitor
|
|
if(m_bReadOnly) {
|
|
// User is not Admin, Disable all of the controls
|
|
EnableWindow(GetDlgItem(m_hwnd, AddButton), FALSE);
|
|
EnableWindow(GetDlgItem(m_hwnd, RemoveButton), FALSE);
|
|
EnableWindow(GetDlgItem(m_hwnd, DefaultButton), FALSE);
|
|
|
|
// EnableWindow(m_hwndList, FALSE);
|
|
|
|
// Set the focus to the OK button
|
|
SetFocus(GetDlgItem(m_hwndSheet, IDOK));
|
|
return FALSE; // Because we moved the focus!
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
// Command processing- we never let them click into the edit control, to prevent them from editing it.
|
|
|
|
BOOL CMonitorProfileManagement::OnCommand(WORD wNotifyCode, WORD wid, HWND hwndCtl) {
|
|
switch (wNotifyCode) {
|
|
case LBN_DBLCLK: {
|
|
// Retrieve the ID of the new default profile
|
|
// only accept dblclk changes if the dialog
|
|
// is not read only - i.e. user is admin
|
|
if (!m_bReadOnly) {
|
|
int id = (int)SendMessage(m_hwndList, LB_GETCURSEL, 0, 0);
|
|
m_uDefault = (unsigned)SendMessage(m_hwndList, LB_GETITEMDATA, id, 0);
|
|
|
|
// Change has been made, update the list
|
|
FillList(DEVLIST_CHANGED);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
case BN_CLICKED:
|
|
switch (wid) {
|
|
case RemoveButton: {
|
|
// Make sure we've tracked the default profile correctly when a profile is removed.
|
|
// All cases break, because we then want the base class to process this message.
|
|
|
|
LRESULT id = SendMessage(m_hwndList, LB_GETCURSEL, 0, 0);
|
|
|
|
unsigned uTarget = (unsigned)SendMessage(m_hwndList, LB_GETITEMDATA, id, 0);
|
|
|
|
if (uTarget > m_uDefault || m_uDefault == (unsigned) -1)
|
|
break; // Nothing here to worry about
|
|
|
|
if (m_uDefault == uTarget) {
|
|
if (CGlobals::ReportEx(AskRemoveDefault, m_hwnd, FALSE, MB_YESNO|MB_ICONEXCLAMATION,0) == IDYES) {
|
|
|
|
// The default has been deleted- the profile
|
|
// at the top of monitor profile list will be
|
|
// made the default profile, if we have
|
|
|
|
LRESULT itemCount = SendMessage(m_hwndList, LB_GETCOUNT, 0, 0);
|
|
|
|
if ((itemCount != LB_ERR) && (itemCount > 1)) {
|
|
m_uDefault = (unsigned)SendMessage(m_hwndList, LB_GETITEMDATA, 0, 0);
|
|
if (m_uDefault == uTarget) {
|
|
m_uDefault = (unsigned)SendMessage(m_hwndList, LB_GETITEMDATA, 1, 0);
|
|
}
|
|
} else {
|
|
m_uDefault = -1;
|
|
}
|
|
|
|
break;
|
|
} else {
|
|
return TRUE; // operation cancelled.
|
|
}
|
|
}
|
|
|
|
if (uTarget < m_cpaProfile.Count())
|
|
break; // We're fine
|
|
|
|
// Must be an added profile below us in the list was
|
|
// zapped- we need to decrement ourselves.
|
|
|
|
m_uDefault--;
|
|
break;
|
|
}
|
|
case DefaultButton: {
|
|
LRESULT id = SendMessage(m_hwndList, LB_GETCURSEL, 0, 0);
|
|
|
|
m_uDefault = (unsigned)SendMessage(m_hwndList, LB_GETITEMDATA, id, 0);
|
|
|
|
// Change has been made, update the list
|
|
FillList(DEVLIST_CHANGED);
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
// Deliberate fall-through (use a break if you add a case here)
|
|
}
|
|
|
|
// Use common command handling if not handled above
|
|
return CDeviceProfileManagement::OnCommand(wNotifyCode, wid, hwndCtl);
|
|
}
|
|
|
|
// Property Sheet notification processing
|
|
|
|
BOOL CMonitorProfileManagement::OnNotify(int idCtrl, LPNMHDR pnmh)
|
|
{
|
|
switch (pnmh -> code) {
|
|
case PSN_APPLY: {
|
|
DisableApplyButton();
|
|
|
|
// If nothing changed, nothing need to do.
|
|
if (!SettingChanged())
|
|
return TRUE;
|
|
|
|
// If the user hasn't selected a default, and we have associated profiles, then we can't allow this.
|
|
|
|
// 03-08-1997 A-RobKj Fix for Memphis RAID 18416- if there's only one profile left, then it must be the default.
|
|
|
|
if (m_uDefault == (unsigned) -1 && (m_cpaAdds.Count() + m_cpaProfile.Count() - m_cuaRemovals.Count()) > 1) {
|
|
CGlobals::Report(NoDefaultProfile, m_hwndSheet);
|
|
SetWindowLongPtr(m_hwnd, DWLP_MSGRESULT, PSNRET_INVALID_NOCHANGEPAGE);
|
|
break;
|
|
}
|
|
|
|
// !!! This behavior is hardly depend on EnumColorProfiles() API !!!
|
|
|
|
// OK, if the default profile has changed, we have to delete default profile associations, and do the association for default profile in "last".
|
|
|
|
// Let the base class handle the cases where the default hasn't changed, or we started with no profiles and still have none.
|
|
|
|
// 03-08-1997 Sleazy code note.
|
|
// The case where no default is selected but only one is assigned will now fall here.
|
|
// Since the default happens to be the "last", and there only is one, letting
|
|
// the base class handle it is not a problem.
|
|
// The list filling code will take care of the rest for us.
|
|
|
|
if (m_uDefault == (unsigned) -1) break;
|
|
|
|
// Remove default first (if default is associated), then associate later.
|
|
|
|
if (m_uDefault < m_cpaProfile.Count())
|
|
m_cpaProfile[m_uDefault] -> Dissociate(m_csDevice);
|
|
|
|
// Now do the other removals (actually just dissociations)
|
|
|
|
for (unsigned u = 0; u < m_cuaRemovals.Count(); u++) {
|
|
m_cpaProfile[m_cuaRemovals[u]] -> Dissociate(m_csDevice);
|
|
}
|
|
|
|
// Add in the new ones
|
|
|
|
for (u = 0; u < m_cpaAdds.Count(); u++) {
|
|
if (m_uDefault >= m_cpaProfile.Count())
|
|
if (u == (m_uDefault - m_cpaProfile.Count()))
|
|
continue; // this is default, will be done later
|
|
|
|
// OK, add it back in...
|
|
m_cpaAdds[u] -> Associate(m_csDevice);
|
|
}
|
|
|
|
// Finally, associate default profile.
|
|
|
|
if (m_uDefault < m_cpaProfile.Count())
|
|
m_cpaProfile[m_uDefault] -> Associate(m_csDevice);
|
|
else
|
|
m_cpaAdds[m_uDefault - m_cpaProfile.Count()] -> Associate(m_csDevice);
|
|
|
|
// Update the various working structures...
|
|
|
|
InitList();
|
|
FillList();
|
|
|
|
SetWindowLongPtr(m_hwnd, DWLP_MSGRESULT, PSNRET_NOERROR);
|
|
|
|
// Now, we have updated settings.
|
|
SettingChanged(FALSE);
|
|
}
|
|
}
|
|
|
|
// Let the base class handle everything else
|
|
return CDeviceProfileManagement::OnNotify(idCtrl, pnmh);
|
|
}
|
|
|
|
// Context-sensitive help handler
|
|
|
|
BOOL CMonitorProfileManagement::OnHelp(LPHELPINFO pHelp)
|
|
{
|
|
if (pHelp->iContextType == HELPINFO_WINDOW) {
|
|
WinHelp((HWND) pHelp->hItemHandle, WINDOWS_HELP_FILE, HELP_WM_HELP, (ULONG_PTR) (LPSTR) MonitorUIHelpIds);
|
|
}
|
|
|
|
return (TRUE);
|
|
}
|
|
|
|
|
|
BOOL CMonitorProfileManagement::OnContextMenu(HWND hwnd)
|
|
{
|
|
WinHelp(hwnd, WINDOWS_HELP_FILE, HELP_CONTEXTMENU, (ULONG_PTR) (LPSTR) MonitorUIHelpIds);
|
|
return (TRUE);
|
|
} |