Windows2003-3790/admin/snapin/sendcmsg/dialogs.cpp
2020-09-30 16:53:55 +02:00

1142 lines
39 KiB
C++

//+---------------------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1997-2002.
//
// File: Dialogs.cpp
//
// Contents:
//
//----------------------------------------------------------------------------
/////////////////////////////////////////////////////////////////////
// Dialogs.cpp
//
// DlgProc for Send Console Message Snapin.
//
// HISTORY
// 4-Aug-97 t-danm Creation.
// 13 Feb 2001 bryanwal Use object picker instead of add recipients
// dialog
/////////////////////////////////////////////////////////////////////
#include "stdafx.h"
#include <strsafe.h>
#include <objsel.h>
#include "debug.h"
#include "util.h"
#include "dialogs.h"
#include "resource.h"
#include <htmlhelp.h> //<mmc.h>
#if 1
#define ThreadTrace0(sz) Trace0(sz)
#define ThreadTrace1(sz, p1) Trace1(sz, p1)
#else
#define ThreadTrace0(sz)
#define ThreadTrace1(sz, p1)
#endif
const PCWSTR CONTEXT_HELP_FILE = L"sendcmsg.hlp";
const PCWSTR HTML_HELP_FILE = L"sendcmsg.chm";
// Register clipboard formats used by the Send Console Message
UINT g_cfSendConsoleMessageText = ::RegisterClipboardFormat(_T("mmc.sendcmsg.MessageText"));
UINT g_cfSendConsoleMessageRecipients = ::RegisterClipboardFormat(_T("mmc.sendcmsg.MessageRecipients"));
enum
{
iImageComputer = 0, // Generic image of a computer
iImageComputerOK,
iImageComputerError
};
// Maximum length of a recipient (machine name)
const int cchRECIPIENT_NAME_MAX = MAX_PATH;
enum
{
COL_NAME = 0,
COL_RESULT,
NUM_COLS // must be last
};
/////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////////////////////////////////////////////
// Generic Computer Picker
///////////////////////////////////////////////////////////////////////////////
//+--------------------------------------------------------------------------
//
// Function: InitObjectPickerForComputers
//
// Synopsis: Call IDsObjectPicker::Initialize with arguments that will
// set it to allow the user to pick a single computer object.
//
// Arguments: [pDsObjectPicker] - object picker interface instance
//
// Returns: Result of calling IDsObjectPicker::Initialize.
//
// History: 10-14-1998 DavidMun Created
//
//---------------------------------------------------------------------------
HRESULT InitObjectPickerForComputers(IDsObjectPicker *pDsObjectPicker)
{
if ( !pDsObjectPicker )
return E_POINTER;
//
// Prepare to initialize the object picker.
// Set up the array of scope initializer structures.
//
static const int SCOPE_INIT_COUNT = 2;
DSOP_SCOPE_INIT_INFO aScopeInit[SCOPE_INIT_COUNT];
ZeroMemory(aScopeInit, sizeof(aScopeInit));
//
// 127399: JonN 10/30/00 JOINED_DOMAIN should be starting scope
//
aScopeInit[0].cbSize = sizeof(DSOP_SCOPE_INIT_INFO);
aScopeInit[0].flType = DSOP_SCOPE_TYPE_UPLEVEL_JOINED_DOMAIN
| DSOP_SCOPE_TYPE_DOWNLEVEL_JOINED_DOMAIN;
aScopeInit[0].flScope = DSOP_SCOPE_FLAG_STARTING_SCOPE;
aScopeInit[0].FilterFlags.Uplevel.flBothModes = DSOP_FILTER_COMPUTERS;
aScopeInit[0].FilterFlags.flDownlevel = DSOP_DOWNLEVEL_FILTER_COMPUTERS;
aScopeInit[1].cbSize = sizeof(DSOP_SCOPE_INIT_INFO);
aScopeInit[1].flType = DSOP_SCOPE_TYPE_ENTERPRISE_DOMAIN
| DSOP_SCOPE_TYPE_GLOBAL_CATALOG
| DSOP_SCOPE_TYPE_EXTERNAL_UPLEVEL_DOMAIN
| DSOP_SCOPE_TYPE_EXTERNAL_DOWNLEVEL_DOMAIN
| DSOP_SCOPE_TYPE_WORKGROUP
| DSOP_SCOPE_TYPE_USER_ENTERED_UPLEVEL_SCOPE
| DSOP_SCOPE_TYPE_USER_ENTERED_DOWNLEVEL_SCOPE;
aScopeInit[1].FilterFlags.Uplevel.flBothModes = DSOP_FILTER_COMPUTERS;
aScopeInit[1].FilterFlags.flDownlevel = DSOP_DOWNLEVEL_FILTER_COMPUTERS;
//
// Put the scope init array into the object picker init array
//
DSOP_INIT_INFO initInfo;
ZeroMemory(&initInfo, sizeof(initInfo));
initInfo.cbSize = sizeof(initInfo);
initInfo.pwzTargetComputer = NULL; // NULL == local machine
initInfo.cDsScopeInfos = SCOPE_INIT_COUNT;
initInfo.aDsScopeInfos = aScopeInit;
initInfo.cAttributesToFetch = 1;
static PCWSTR pwszDnsHostName = L"dNSHostName";
initInfo.apwzAttributeNames = &pwszDnsHostName;
//
// Note object picker makes its own copy of initInfo. Also note
// that Initialize may be called multiple times, last call wins.
//
return pDsObjectPicker->Initialize(&initInfo);
}
//+--------------------------------------------------------------------------
//
// Function: ProcessSelectedObjects
//
// Synopsis: Retrieve the list of selected items from the data object
// created by the object picker and print out each one.
//
// Arguments: [pdo] - data object returned by object picker
//
// History: 10-14-1998 DavidMun Created
//
//---------------------------------------------------------------------------
HRESULT ProcessSelectedObjects(IDataObject *pdo, PWSTR computerName, int cchLen)
{
Assert (pdo && computerName);
if ( !pdo || !computerName)
return E_POINTER;
HRESULT hr = S_OK;
static UINT g_cfDsObjectPicker =
RegisterClipboardFormat(CFSTR_DSOP_DS_SELECTION_LIST);
STGMEDIUM stgmedium =
{
TYMED_HGLOBAL,
NULL,
NULL
};
FORMATETC formatetc =
{
(CLIPFORMAT)g_cfDsObjectPicker,
NULL,
DVASPECT_CONTENT,
-1,
TYMED_HGLOBAL
};
bool fGotStgMedium = false;
do
{
hr = pdo->GetData(&formatetc, &stgmedium);
if ( SUCCEEDED (hr) )
{
fGotStgMedium = true;
PDS_SELECTION_LIST pDsSelList =
(PDS_SELECTION_LIST) GlobalLock(stgmedium.hGlobal);
if (!pDsSelList)
{
hr = HRESULT_FROM_WIN32 (GetLastError());
break;
}
Assert (1 == pDsSelList->cItems);
if ( 1 == pDsSelList->cItems )
{
PDS_SELECTION psel = &(pDsSelList->aDsSelection[0]);
VARIANT* pvarDnsName = &(psel->pvarFetchedAttributes[0]);
if ( NULL == pvarDnsName
|| VT_BSTR != pvarDnsName->vt
|| NULL == pvarDnsName->bstrVal
|| L'\0' == (pvarDnsName->bstrVal)[0] )
{
// security review 3/1/2002 BryanWal
// ISSUE - possible non-null termination - convert to strsafe
// NTRAID# Bug9 560859 security: SendCMsg: possible non-null termination of computer name
wcsncpy (computerName, psel->pwzName, cchLen);
}
else
{
// security review 3/1/2002 BryanWal
// ISSUE - possible non-null termination - convert to strsafe
// NTRAID# Bug9 560859 security: SendCMsg: possible non-null termination of computer name
wcsncpy (computerName, pvarDnsName->bstrVal, cchLen);
}
}
else
hr = E_UNEXPECTED;
GlobalUnlock(stgmedium.hGlobal);
}
} while (0);
if (fGotStgMedium)
{
ReleaseStgMedium(&stgmedium);
}
return hr;
}
///////////////////////////////////////////////////////////////////////////////
// Generic method for launching a single-select computer picker
//
// Paremeters:
// hwndParent (IN) - window handle of parent window
// computerName (OUT) - computer name returned
//
// Returns S_OK if everything succeeded, S_FALSE if user pressed "Cancel"
//
//////////////////////////////////////////////////////////////////////////////
HRESULT ComputerNameFromObjectPicker (HWND hwndParent, PWSTR computerName, int cchLen)
{
Assert (computerName);
if ( !computerName )
return E_POINTER;
//
// Create an instance of the object picker. The implementation in
// objsel.dll is apartment model.
//
CComPtr<IDsObjectPicker> spDsObjectPicker;
// security review 3/1/2002 BryanWal ok
HRESULT hr = CoCreateInstance(CLSID_DsObjectPicker,
NULL,
CLSCTX_INPROC_SERVER,
IID_IDsObjectPicker,
(void **) &spDsObjectPicker);
if ( SUCCEEDED (hr) )
{
Assert(!!spDsObjectPicker);
//
// Initialize the object picker to choose computers
//
hr = InitObjectPickerForComputers(spDsObjectPicker);
if ( SUCCEEDED (hr) )
{
//
// Now pick a computer
//
CComPtr<IDataObject> spDataObject;
hr = spDsObjectPicker->InvokeDialog(hwndParent, &spDataObject);
if ( S_OK == hr )
{
Assert(!!spDataObject);
hr = ProcessSelectedObjects(spDataObject, computerName, cchLen);
}
}
}
return hr;
}
/////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////
CSendConsoleMessageDlg::CSendConsoleMessageDlg()
: m_cRefCount (0),
m_hImageList (0),
m_hdlg (0),
m_hwndEditMessageText (0),
m_hwndListviewRecipients (0)
{
m_DispatchInfo.pargbItemStatus = NULL;
// security review 3/1/2002 BryanWal
// ISSUE - can raise a STATUS_NO_MEMORY exception. consider pre-allocating at DLL_PROCESS_ATTACH
// NTRAID Bug9 565939 SendCMsg: InitializeCriticalSection throws uncaught exception
InitializeCriticalSection(OUT &m_DispatchInfo.cs);
}
CSendConsoleMessageDlg::~CSendConsoleMessageDlg()
{
ThreadTrace0("Destroying CSendConsoleMessageDlg object.\n");
Assert(m_hdlg == NULL);
delete m_DispatchInfo.pargbItemStatus;
DeleteCriticalSection(IN &m_DispatchInfo.cs);
}
/////////////////////////////////////////////////////////////////////
void CSendConsoleMessageDlg::AddRef()
{
// ISSUE - use interlocked increment
// security review 3/1/2002 BryanWal
// NTRAID# Bug9 561315 Security: SendCMsg: Replace critical sections with interlocked_increment
EnterCriticalSection(INOUT &m_DispatchInfo.cs);
Assert(m_cRefCount >= 0);
Assert(HIWORD(m_cRefCount) == 0);
m_cRefCount++;
LeaveCriticalSection(INOUT &m_DispatchInfo.cs);
}
/////////////////////////////////////////////////////////////////////
void CSendConsoleMessageDlg::Release()
{
// Security Review 3/1/2002 BryanWal ok
EnterCriticalSection(INOUT &m_DispatchInfo.cs);
Assert(HIWORD(m_cRefCount) == 0);
m_cRefCount--;
BOOL fDeleteObject = (m_cRefCount <= 0);
if (m_hdlg != NULL)
{
Assert(IsWindow(m_hdlg));
// Cause the UI to refresh
PostMessage(m_hdlg, WM_COMMAND, MAKEWPARAM(IDC_EDIT_MESSAGE_TEXT, EN_CHANGE), 0);
}
LeaveCriticalSection(INOUT &m_DispatchInfo.cs);
if (fDeleteObject)
delete this;
}
/////////////////////////////////////////////////////////////////////
void CSendConsoleMessageDlg::OnInitDialog(HWND hdlg, IDataObject * pDataObject)
{
Assert(IsWindow(hdlg));
Assert(pDataObject != NULL);
if ( !IsWindow (hdlg) || ! pDataObject )
return;
m_hdlg = hdlg;
m_hwndEditMessageText = GetDlgItem(m_hdlg, IDC_EDIT_MESSAGE_TEXT);
m_hwndListviewRecipients = GetDlgItem(m_hdlg, IDC_LIST_RECIPIENTS);
Assert(::IsWindow(m_hwndEditMessageText));
Assert(::IsWindow(m_hwndListviewRecipients));
WCHAR * pawszMessage = NULL;
(void) HrExtractDataAlloc(IN pDataObject, g_cfSendConsoleMessageText, OUT (PVOID *)&pawszMessage);
// Set the initial message text
if ( pawszMessage )
{
SetWindowTextW(m_hwndEditMessageText, pawszMessage);
GlobalFree(pawszMessage);
}
SendMessage(m_hwndEditMessageText, EM_SETSEL, 0, 0);
SetFocus(m_hwndEditMessageText);
Assert(m_hImageList == NULL);
m_hImageList = ImageList_LoadImage(
g_hInstance,
MAKEINTRESOURCE(IDB_BITMAP_COMPUTER),
16, 3, RGB(255, 0, 255),
IMAGE_BITMAP, 0);
Report(m_hImageList != NULL);
ListView_SetImageList(m_hwndListviewRecipients, m_hImageList, LVSIL_SMALL);
// Set up columns in list view
int colWidths[NUM_COLS] = {200, 200};
LVCOLUMN lvColumn;
WCHAR szColumnText[128];
::ZeroMemory (&lvColumn, sizeof (lvColumn));
lvColumn.mask = LVCF_WIDTH | LVCF_TEXT;
lvColumn.cx = colWidths[COL_NAME];
CchLoadString (IDS_RECIPIENT, OUT szColumnText, LENGTH(szColumnText));
lvColumn.pszText = szColumnText;
int nCol = ListView_InsertColumn (m_hwndListviewRecipients, COL_NAME, &lvColumn);
Assert (-1 != nCol);
lvColumn.cx = colWidths[COL_RESULT];
CchLoadString (IDS_MESSAGE_STATUS, OUT szColumnText, LENGTH(szColumnText));
lvColumn.pszText = szColumnText;
nCol = ListView_InsertColumn (m_hwndListviewRecipients, COL_RESULT, &lvColumn);
Assert (-1 != nCol);
if ( -1 != nCol )
{
// Make column fill remaining space
ListView_SetColumnWidth (m_hwndListviewRecipients, COL_RESULT,
LVSCW_AUTOSIZE_USEHEADER);
}
// Get the list of recipients
WCHAR * pagrwszRecipients = NULL;
(void)HrExtractDataAlloc(IN pDataObject, g_cfSendConsoleMessageRecipients, OUT (PVOID *)&pagrwszRecipients);
if (pagrwszRecipients == NULL)
{
UpdateUI();
return;
}
// Add the recipients to the listview
const WCHAR * pszRecipient = pagrwszRecipients;
while (*pszRecipient != '\0')
{
// Strip off leading "\\" if present.
// security review 3/1/2002 BryanWal ok
if ( !_wcsnicmp (pszRecipient, L"\\\\", 2) )
{
pszRecipient+= 2;
}
AddRecipient(pszRecipient);
while(*pszRecipient++ != '\0')
; // Skip until the next string
} // while
// NTRAID# 213370 [SENDCMSG] Accessibility - Main dialog tab stop on
// Recipients listview has no visible focus indicator until object is
// selected
int nIndex = ListView_GetTopIndex (m_hwndListviewRecipients);
ListView_SetItemState (m_hwndListviewRecipients, nIndex, LVIS_FOCUSED,
LVIS_FOCUSED);
GlobalFree(pagrwszRecipients);
UpdateUI();
} // CSendConsoleMessageDlg::OnInitDialog()
/////////////////////////////////////////////////////////////////////
void CSendConsoleMessageDlg::OnOK()
{
Assert(m_cRefCount == 1 && "There is already another thread running.");
m_DispatchInfo.status = e_statusDlgInit;
m_DispatchInfo.cErrors = 0;
delete m_DispatchInfo.pargbItemStatus;
m_DispatchInfo.pargbItemStatus = NULL;
(void)DoDialogBox(IDD_DISPATCH_MESSAGES, m_hdlg,
DlgProcDispatchMessageToRecipients, (LPARAM)this);
if (m_DispatchInfo.cErrors == 0 && m_DispatchInfo.status == e_statusDlgCompleted)
{
// No problems dispatching the message to recipients
EndDialog(m_hdlg, TRUE); // Close the dialog
return;
}
Assert(IsWindow(m_hwndListviewRecipients));
ListView_UnselectAllItems(m_hwndListviewRecipients);
if (m_DispatchInfo.cErrors > 0)
{
DoMessageBox(m_hdlg, IDS_ERR_CANNOT_SEND_TO_ALL_RECIPIENTS);
}
// We did not finished the job, so display the status to the UI
if (m_DispatchInfo.pargbItemStatus == NULL)
{
// The progress was unable to allocate memory for the status
Trace0("CSendConsoleMessageDlg::OnOK() - Out of memory.\n");
return;
}
// Remove all the successful items, leaving only the failed targets and
// the unsent targets (in the event the user pressed Cancel).
int iItem = ListView_GetItemCount (m_hwndListviewRecipients);
iItem--;
const BYTE * pb = m_DispatchInfo.pargbItemStatus + iItem;
for (; iItem >= 0 && pb >= m_DispatchInfo.pargbItemStatus;
pb--, iItem--)
{
if ( *pb == iImageComputerOK )
VERIFY (ListView_DeleteItem (m_hwndListviewRecipients, iItem));
}
} // CSendConsoleMessageDlg::OnOK()
/////////////////////////////////////////////////////////////////////
void CSendConsoleMessageDlg::DispatchMessageToRecipients()
{
const size_t FORMAT_BUF_LEN = 128;
const int cRecipients = ListView_GetItemCount(m_hwndListviewRecipients);
WCHAR szT[FORMAT_BUF_LEN + cchRECIPIENT_NAME_MAX];
WCHAR szFmtStaticRecipient[FORMAT_BUF_LEN]; // "Sending console message to %s..."
WCHAR szFmtStaticMessageOf[FORMAT_BUF_LEN]; // "Sending message %d of %d."
WCHAR szFmtStaticTotalErrors[FORMAT_BUF_LEN]; // "Total errors encountered\t%d."
GetWindowText(m_DispatchInfo.hctlStaticRecipient, OUT szFmtStaticRecipient, LENGTH(szFmtStaticRecipient));
GetWindowText(m_DispatchInfo.hctlStaticMessageOf, szFmtStaticMessageOf, LENGTH(szFmtStaticMessageOf));
GetWindowText(m_DispatchInfo.hctlStaticErrors, OUT szFmtStaticTotalErrors, LENGTH(szFmtStaticTotalErrors));
SendMessage(m_DispatchInfo.hctlProgressBar, PBM_SETRANGE, 0, MAKELPARAM(0, cRecipients));
//
// Set the image of each recipient to normal computer
//
ListView_UnselectAllItems(m_hwndListviewRecipients);
for (int i = 0; i < cRecipients; i++)
{
ListView_SetItemImage(m_hwndListviewRecipients, i, iImageComputer);
ListView_SetItemText(m_hwndListviewRecipients, i, COL_RESULT, L"");
}
UpdateUI(); // Update the other UI controls (especially OK button)
//
// Get the text from the edit control
//
int cchMessage = GetWindowTextLength(m_hwndEditMessageText) + 1;
WCHAR * pawszMessage = new WCHAR[cchMessage];
if (pawszMessage != NULL)
{
// security review 3/1/2002 BryanWal ok - cchMessage includes null terminator
GetWindowTextW(m_hwndEditMessageText, OUT pawszMessage, cchMessage);
}
else
{
cchMessage = 0;
Trace0("Unable to allocate memory for message.\n");
}
WCHAR wszRecipient[cchRECIPIENT_NAME_MAX];
LV_ITEMW lvItem;
::ZeroMemory (&lvItem, sizeof(lvItem));
lvItem.iItem = 0;
lvItem.iSubItem = 0;
lvItem.pszText = wszRecipient;
lvItem.cchTextMax = LENGTH(wszRecipient);
Assert(m_DispatchInfo.pargbItemStatus == NULL && "Memory Leak");
m_DispatchInfo.pargbItemStatus = new BYTE[cRecipients+1];
if (m_DispatchInfo.pargbItemStatus != NULL)
{
// security review 3/1/2002 BryanWal ok
memset(OUT m_DispatchInfo.pargbItemStatus, iImageComputer, cRecipients+1);
}
else
{
Trace0("Unable to allocate memory for listview item status.\n");
}
Assert(m_DispatchInfo.status == e_statusDlgInit);
m_DispatchInfo.status = e_statusDlgDispatching; // Allow the user to cancel the dialog
WCHAR szFailure[128];
CchLoadString(IDS_MESSAGE_COULD_NOT_BE_SENT, OUT szFailure,
LENGTH(szFailure));
for (i = 0; i < cRecipients; i++)
{
ThreadTrace1("Sending message to recipient %d.\n", i + 1);
// security review 3/1/2002 BryanWal ok
EnterCriticalSection(INOUT &m_DispatchInfo.cs);
if (m_DispatchInfo.status == e_statusUserCancel)
{
ThreadTrace0("DispatchMessageToRecipients() - Aborting loop @1...\n");
LeaveCriticalSection(INOUT &m_DispatchInfo.cs);
break;
}
ListView_SelectItem(m_hwndListviewRecipients, i);
ListView_EnsureVisible(m_hwndListviewRecipients, i, FALSE);
lvItem.iItem = i;
wszRecipient[0] = '\0';
// Get the recipient name
SendMessage(m_hwndListviewRecipients, LVM_GETITEMTEXTW, i, OUT (LPARAM)&lvItem);
if (m_DispatchInfo.pargbItemStatus != NULL)
m_DispatchInfo.pargbItemStatus[i] = iImageComputerError;
// security review 3/1/2002 BryanWal
// Issue: convert to strsafe - possible buffer overflow because of static allocation
HRESULT hr = ::StringCchPrintf (OUT szT, sizeof (szT)/sizeof (szT[0]), szFmtStaticRecipient, wszRecipient);
Assert (SUCCEEDED (hr));
if ( FAILED (hr) )
continue;
SetWindowTextW(m_DispatchInfo.hctlStaticRecipient, szT);
// security review 3/1/2002 BryanWal
// Issue: convert to strsafe - possible buffer overflow because of static allocation
hr = ::StringCchPrintf (OUT szT, sizeof (szT)/sizeof (szT[0]), szFmtStaticMessageOf, i + 1, cRecipients);
Assert (SUCCEEDED (hr));
if ( FAILED (hr) )
continue;
SetWindowText(m_DispatchInfo.hctlStaticMessageOf, szT);
switch ( m_DispatchInfo.cErrors )
{
case 0:
break;
case 1:
::ShowWindow (m_DispatchInfo.hctlStaticErrors, SW_SHOW);
{
WCHAR sz1NotSet[128];
CchLoadString(IDS_1_RECIPIENT_NOT_CONTACTED, OUT sz1NotSet,
LENGTH(sz1NotSet));
SetWindowText(m_DispatchInfo.hctlStaticErrors, sz1NotSet);
}
break;
default:
// security review 3/1/2002 BryanWal
// ISSUE - convert to strsafe - possible buffer overflow because of static allocation
hr = ::StringCchPrintf (OUT szT, sizeof (szT)/sizeof (szT[0]), szFmtStaticTotalErrors, m_DispatchInfo.cErrors);
Assert (SUCCEEDED (hr));
if ( SUCCEEDED (hr) )
SetWindowText(m_DispatchInfo.hctlStaticErrors, szT);
break;
}
LeaveCriticalSection(INOUT &m_DispatchInfo.cs);
// Send the message to the recipient (ie, computer)
NET_API_STATUS err;
err = ::NetMessageBufferSend(
NULL,
wszRecipient,
NULL,
(BYTE *)pawszMessage,
cchMessage * sizeof(WCHAR));
int iImage = iImageComputerOK;
if (err != ERROR_SUCCESS)
{
Trace3("Error sending message to recipient %ws. err=%d (0x%X).\n", wszRecipient, err, err);
m_DispatchInfo.cErrors++;
iImage = iImageComputerError;
}
if (m_DispatchInfo.pargbItemStatus != NULL)
m_DispatchInfo.pargbItemStatus[i] = (BYTE)iImage;
// security review 3/1/2002 BryanWal - ok - nothing in here throws an exception
EnterCriticalSection(INOUT &m_DispatchInfo.cs);
if (m_DispatchInfo.status == e_statusUserCancel)
{
ThreadTrace0("DispatchMessageToRecipients() - Aborting loop @2...\n");
LeaveCriticalSection(INOUT &m_DispatchInfo.cs);
break;
}
//
// Update the listview
//
ListView_UnselectItem(m_hwndListviewRecipients, i);
ListView_SetItemImage(m_hwndListviewRecipients, i, iImage);
if ( iImage == iImageComputerError )
ListView_SetItemText(m_hwndListviewRecipients, i, COL_RESULT,
szFailure);
//
// Update the progress dialog
//
SendMessage(m_DispatchInfo.hctlProgressBar, PBM_SETPOS, i + 1, 0);
LeaveCriticalSection(INOUT &m_DispatchInfo.cs);
} // for
delete [] pawszMessage;
Sleep(500);
// security review 3/1/2002 BryanWal ok
EnterCriticalSection(INOUT &m_DispatchInfo.cs);
if (m_DispatchInfo.status != e_statusUserCancel)
{
// We are done dispatching the message to all the recipients
// and the user did not canceled the operation.
m_DispatchInfo.status = e_statusDlgCompleted;
Assert(IsWindow(m_DispatchInfo.hdlg));
EndDialog(m_DispatchInfo.hdlg, TRUE); // Gracefully close the dialog
}
LeaveCriticalSection(INOUT &m_DispatchInfo.cs);
} // CSendConsoleMessageDlg::DispatchMessageToRecipients()
/////////////////////////////////////////////////////////////////////
// Add a recipient to the listview control
//
// Return the index of the inserted item.
//
int CSendConsoleMessageDlg::AddRecipient(
PCWSTR pszRecipient, // IN: Machine name
BOOL fSelectItem) // TRUE => Select the item that is inserted
{
Assert(pszRecipient != NULL);
// NTRAID# 498210 [Send Console Message] User can add the same computer to
// the Recipients listbox multiple times, sending multiple messages
LVFINDINFO lvfi;
::ZeroMemory (&lvfi, sizeof (lvfi));
lvfi.flags = LVFI_STRING;
lvfi.psz = const_cast<WCHAR *>(pszRecipient);
if ( -1 == ListView_FindItem (m_hwndListviewRecipients, -1, &lvfi) )
{
LV_ITEM lvItem;
::ZeroMemory (&lvItem, sizeof(lvItem));
lvItem.mask = LVIF_TEXT | LVIF_IMAGE;
lvItem.iSubItem = 0;
lvItem.iImage = iImageComputer;
lvItem.pszText = const_cast<WCHAR *>(pszRecipient);
if (fSelectItem)
{
lvItem.mask = LVIF_TEXT | LVIF_IMAGE |LVIF_STATE;
lvItem.state = LVIS_SELECTED;
}
return ListView_InsertItem(m_hwndListviewRecipients, IN &lvItem);
}
else
return -1;
} // CSendConsoleMessageDlg::AddRecipient()
/////////////////////////////////////////////////////////////////////
LRESULT CSendConsoleMessageDlg::OnNotify(NMHDR * pNmHdr)
{
Assert(pNmHdr != NULL);
switch (pNmHdr->code)
{
case LVN_ENDLABELEDIT:
{
WCHAR * pszText = ((LV_DISPINFO *)pNmHdr)->item.pszText;
if (pszText == NULL)
break; // User canceled editing
// HACK: Modifying a string which I'm not sure where it is allocated
(void)FTrimString(INOUT pszText);
// Check out if there is already another recipient
int iItem = ListView_FindString(m_hwndListviewRecipients, pszText);
if (iItem >= 0)
{
ListView_SelectItem(m_hwndListviewRecipients, iItem);
DoMessageBox(m_hdlg, IDS_RECIPIENT_ALREADY_EXISTS);
break;
}
// Otherwise accept the changes
SetWindowLongPtr(m_hdlg, DWLP_MSGRESULT, TRUE);
return TRUE;
}
case LVN_ITEMCHANGED: // Selection changed
UpdateUI();
break;
case LVN_KEYDOWN:
switch (((LV_KEYDOWN *)pNmHdr)->wVKey)
{
case VK_INSERT:
SendMessage(m_hdlg, WM_COMMAND, IDC_BUTTON_ADD_RECIPIENT, 0);
break;
case VK_DELETE:
SendMessage(m_hdlg, WM_COMMAND, IDC_BUTTON_REMOVE_RECIPIENT, 0);
break;
} // switch
break;
case NM_CLICK:
UpdateUI();
break;
case NM_DBLCLK:
UpdateUI();
break;
} // switch
return 0;
} // CSendConsoleMessageDlg::OnNotify()
/////////////////////////////////////////////////////////////////////
void CSendConsoleMessageDlg::EnableDlgItem(INT nIdDlgItem, BOOL fEnable)
{
Assert(::IsWindow(::GetDlgItem(m_hdlg, nIdDlgItem)));
::EnableWindow(::GetDlgItem(m_hdlg, nIdDlgItem), fEnable);
}
/////////////////////////////////////////////////////////////////////
void CSendConsoleMessageDlg::UpdateUI()
{
Assert(m_cRefCount > 0);
int cchMessage = GetWindowTextLength(m_hwndEditMessageText);
int cItems = ListView_GetItemCount(m_hwndListviewRecipients);
EnableDlgItem(IDOK, (cchMessage > 0) && (cItems > 0) && (m_cRefCount == 1));
int iItemSelected = ListView_GetSelectedItem(m_hwndListviewRecipients);
EnableDlgItem(IDC_BUTTON_REMOVE_RECIPIENT, iItemSelected >= 0);
UpdateWindow(m_hwndListviewRecipients);
} // CSendConsoleMessageDlg::UpdateUI()
/////////////////////////////////////////////////////////////////////
// Dialog proc for the Send Console Message snapin.
//
// USAGE
// DoDialogBox(IDD_SEND_CONSOLE_MESSAGE, ::GetActiveWindow(), CSendConsoleMessageDlg::DlgProc);
//
INT_PTR CALLBACK CSendConsoleMessageDlg::DlgProc(HWND hdlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
CSendConsoleMessageDlg * pThis;
pThis = (CSendConsoleMessageDlg *)::GetWindowLongPtr(hdlg, GWLP_USERDATA);
switch (uMsg)
{
case WM_INITDIALOG:
Assert(pThis == NULL);
if (pThis != NULL)
return FALSE;
pThis = new CSendConsoleMessageDlg;
if (pThis == NULL)
{
Trace0("Unable to allocate CSendConsoleMessageDlg object.\n");
return -1;
}
SetWindowLongPtr(hdlg, GWLP_USERDATA, (LONG_PTR)pThis);
pThis->AddRef();
pThis->OnInitDialog(hdlg, (IDataObject *)lParam);
SendDlgItemMessage (hdlg, IDC_EDIT_MESSAGE_TEXT, EM_LIMITTEXT, 885, 0);
return FALSE;
case WM_NCDESTROY:
ThreadTrace0("CSendConsoleMessageDlg::DlgProc() - WM_NCDESTROY.\n");
// security review 3/1/2002 BryanWal
EnterCriticalSection(INOUT &pThis->m_DispatchInfo.cs);
pThis->m_hdlg = NULL;
LeaveCriticalSection(INOUT &pThis->m_DispatchInfo.cs);
pThis->Release();
break;
case WM_COMMAND:
switch (LOWORD(wParam))
{
case IDOK:
if (HIWORD(wParam) == BN_CLICKED)
{
Assert((HWND)lParam == GetDlgItem(hdlg, IDOK));
pThis->OnOK();
}
break;
case IDCANCEL:
if (HIWORD(wParam) == BN_CLICKED)
{
Assert((HWND)lParam == GetDlgItem(hdlg, IDCANCEL));
EndDialog(hdlg, FALSE);
}
break;
case IDC_EDIT_MESSAGE_TEXT:
if (HIWORD(wParam) == EN_CHANGE)
pThis->UpdateUI();
break;
case IDC_BUTTON_ADD_RECIPIENT:
{
WCHAR szComputerName[MAX_PATH];
// S_FALSE means user pressed "Cancel"
if ( S_OK == ComputerNameFromObjectPicker (hdlg,
szComputerName, MAX_PATH) )
{
pThis->AddRecipient (szComputerName, TRUE);
}
pThis->UpdateUI();
}
break;
case IDC_BUTTON_REMOVE_RECIPIENT:
while (TRUE)
{
// Remove all the selected recipients
int iItem = ListView_GetSelectedItem(pThis->m_hwndListviewRecipients);
if (iItem < 0)
break;
ListView_DeleteItem(pThis->m_hwndListviewRecipients, iItem);
}
::SetFocus(pThis->m_hwndListviewRecipients);
pThis->UpdateUI();
break;
case IDC_BUTTON_ADVANCED:
(void)DoDialogBox(IDD_ADVANCED_MESSAGE_OPTIONS, hdlg, CSendMessageAdvancedOptionsDlg::DlgProc);
break;
} // switch
break;
case WM_NOTIFY:
return pThis->OnNotify((NMHDR *)lParam);
case WM_HELP:
return pThis->OnHelp (lParam, IDD_SEND_CONSOLE_MESSAGE);
default:
return FALSE;
} // switch
return TRUE;
} // CSendConsoleMessageDlg::DlgProc()
/////////////////////////////////////////////////////////////////////
// DlgProcDispatchMessageToRecipients()
//
// Private dialog to indicate the progress while a background
// thread dispatches a message to each recipient.
//
INT_PTR CALLBACK CSendConsoleMessageDlg::DlgProcDispatchMessageToRecipients(HWND hdlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
CSendConsoleMessageDlg * pThis = (CSendConsoleMessageDlg *)::GetWindowLongPtr(hdlg, GWLP_USERDATA);
switch (uMsg)
{
case WM_INITDIALOG:
Assert(pThis == NULL);
if (pThis != NULL)
return FALSE;
pThis = (CSendConsoleMessageDlg *)lParam;
SetWindowLongPtr(hdlg, GWLP_USERDATA, (LONG_PTR)pThis);
Assert(pThis != NULL);
Assert(pThis->m_DispatchInfo.status == e_statusDlgInit);
pThis->m_DispatchInfo.hdlg = hdlg;
pThis->m_DispatchInfo.hctlStaticRecipient = GetDlgItem(hdlg, IDC_STATIC_RECIPIENT);
pThis->m_DispatchInfo.hctlStaticMessageOf = GetDlgItem(hdlg, IDC_STATIC_MESSAGE_OF);
pThis->m_DispatchInfo.hctlStaticErrors = GetDlgItem(hdlg, IDC_STATIC_ERRORS_ENCOUNTERED);
pThis->m_DispatchInfo.hctlProgressBar = GetDlgItem(hdlg, IDC_PROGRESS_MESSAGES);
{
DWORD dwThreadId;
HANDLE hThread = ::CreateThread(NULL, 0,
(LPTHREAD_START_ROUTINE)ThreadProcDispatchMessageToRecipients, pThis, 0, OUT &dwThreadId);
Report(hThread != NULL && "Unable to create thread");
if (hThread != NULL)
{
VERIFY( ::CloseHandle(hThread) );
}
else
{
Trace0("Unable to create thread.\n");
// Prevent a potential deadlock
pThis->m_DispatchInfo.status = e_statusUserCancel; // Pretend the user clicked on cancel
EndDialog(hdlg, FALSE);
}
}
break;
case WM_DESTROY:
// Those variables are set to NULL just in case
pThis->m_DispatchInfo.hdlg = NULL;
pThis->m_DispatchInfo.hctlStaticRecipient = NULL;
pThis->m_DispatchInfo.hctlStaticMessageOf = NULL;
pThis->m_DispatchInfo.hctlStaticErrors = NULL;
pThis->m_DispatchInfo.hctlProgressBar = NULL;
break;
case WM_COMMAND:
if (wParam == IDCANCEL)
{
Trace0("INFO: WM_COMMAND: IDCANCEL: User canceled operation.\n");
BOOL fEndDialog = FALSE;
if (TryEnterCriticalSection(INOUT &pThis->m_DispatchInfo.cs))
{
if (pThis->m_DispatchInfo.status != e_statusDlgInit)
{
pThis->m_DispatchInfo.status = e_statusUserCancel;
fEndDialog = TRUE;
}
LeaveCriticalSection(INOUT &pThis->m_DispatchInfo.cs);
}
if (fEndDialog)
{
EndDialog(hdlg, FALSE);
}
else
{
ThreadTrace0("Critical section already in use. Try again...\n");
PostMessage(hdlg, WM_COMMAND, IDCANCEL, 0);
Sleep(100);
} // if...else
} // if
break;
case WM_HELP:
return pThis->OnHelp (lParam, IDD_DISPATCH_MESSAGES);
default:
return FALSE;
} // switch
return TRUE;
} // CSendConsoleMessageDlg::DlgProcDispatchMessageToRecipients()
/////////////////////////////////////////////////////////////////////
DWORD CSendConsoleMessageDlg::ThreadProcDispatchMessageToRecipients(CSendConsoleMessageDlg * pThis)
{
Assert(pThis != NULL);
pThis->AddRef();
Assert(pThis->m_cRefCount > 1);
pThis->DispatchMessageToRecipients();
pThis->Release();
return 0;
} // CSendConsoleMessageDlg::ThreadProcDispatchMessageToRecipients()
#define IDH_EDIT_MESSAGE_TEXT 900
#define IDH_LIST_RECIPIENTS 901
#define IDH_BUTTON_ADD_RECIPIENT 903
#define IDH_BUTTON_REMOVE_RECIPIENT 904
const DWORD g_aHelpIDs_IDD_SEND_CONSOLE_MESSAGE[]=
{
IDC_EDIT_MESSAGE_TEXT, IDH_EDIT_MESSAGE_TEXT,
IDOK, IDOK,
IDC_LIST_RECIPIENTS, IDH_LIST_RECIPIENTS,
IDC_BUTTON_ADD_RECIPIENT, IDH_BUTTON_ADD_RECIPIENT,
IDC_BUTTON_REMOVE_RECIPIENT, IDH_BUTTON_REMOVE_RECIPIENT,
0, 0
};
BOOL CSendConsoleMessageDlg::OnHelp(LPARAM lParam, int nDlgIDD)
{
const LPHELPINFO pHelpInfo = (LPHELPINFO)lParam;
if (pHelpInfo && pHelpInfo->iContextType == HELPINFO_WINDOW)
{
switch (nDlgIDD)
{
case IDD_SEND_CONSOLE_MESSAGE:
DoSendConsoleMessageContextHelp ((HWND) pHelpInfo->hItemHandle);
break;
}
}
else
HtmlHelpW (NULL, HTML_HELP_FILE, HH_DISPLAY_TOPIC, 0);
return TRUE;
}
void CSendConsoleMessageDlg::DoSendConsoleMessageContextHelp (HWND hWndControl)
{
switch (::GetDlgCtrlID (hWndControl))
{
case IDCANCEL:
case IDC_BUTTON_ADVANCED:
break;
default:
// Display context help for a control
if ( !::WinHelp (
hWndControl,
CONTEXT_HELP_FILE,
HELP_WM_HELP,
(DWORD_PTR) g_aHelpIDs_IDD_SEND_CONSOLE_MESSAGE) )
{
Trace1 ("WinHelp () failed: 0x%x\n", GetLastError ());
}
break;
}
}
/////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////
void CSendMessageAdvancedOptionsDlg::OnInitDialog(HWND hdlg)
{
m_hdlg = hdlg;
m_fSendAutomatedMessage = FALSE;
CheckDlgButton(m_hdlg, IDC_CHECK_SEND_AUTOMATED_MESSAGE, m_fSendAutomatedMessage);
UpdateUI();
}
/////////////////////////////////////////////////////////////////////
void CSendMessageAdvancedOptionsDlg::UpdateUI()
{
static const UINT rgid[] =
{
IDC_STATIC_RESOURCE_NAME,
IDC_EDIT_RESOURCE_NAME,
IDC_STATIC_SHUTDOWN_OCCURS,
IDC_EDIT_SHUTDOWN_OCCURS,
//IDC_SPIN_SHUTDOWN_OCCURS,
IDC_STATIC_SHUTDOWN_OCCURS_UNIT,
IDC_STATIC_RESEND,
IDC_EDIT_RESEND,
//IDC_SPIN_RESEND,
IDC_STATIC_RESEND_UNIT,
IDC_STATIC_RESOURCE_BACK_ONLINE,
IDC_EDIT_RESOURCE_BACK_ONLINE,
};
for (int i = 0; i < LENGTH(rgid); i++)
{
EnableWindow(GetDlgItem(m_hdlg, rgid[i]), m_fSendAutomatedMessage);
}
} // CSendMessageAdvancedOptionsDlg::UpdateUI()
/////////////////////////////////////////////////////////////////////
INT_PTR CALLBACK CSendMessageAdvancedOptionsDlg::DlgProc(HWND hdlg, UINT uMsg, WPARAM wParam, LPARAM /*lParam*/)
{
CSendMessageAdvancedOptionsDlg * pThis;
pThis = (CSendMessageAdvancedOptionsDlg *)GetWindowLongPtr(hdlg, GWLP_USERDATA);
switch (uMsg)
{
case WM_INITDIALOG:
Assert(pThis == NULL);
pThis = new CSendMessageAdvancedOptionsDlg;
if (pThis == NULL)
return -1;
SetWindowLongPtr(hdlg, GWLP_USERDATA, (LONG_PTR)pThis);
pThis->OnInitDialog(hdlg);
break;
case WM_COMMAND:
switch (wParam)
{
case IDOK:
EndDialog(hdlg, TRUE);
break;
case IDCANCEL:
EndDialog(hdlg, FALSE);
break;
case IDC_CHECK_SEND_AUTOMATED_MESSAGE:
pThis->m_fSendAutomatedMessage = IsDlgButtonChecked(hdlg, IDC_CHECK_SEND_AUTOMATED_MESSAGE);
pThis->UpdateUI();
break;
} // switch
break;
default:
return FALSE;
} // switch
return TRUE;
} // CSendMessageAdvancedOptionsDlg::DlgProc()
BOOL CSendMessageAdvancedOptionsDlg::OnHelp(LPARAM /*lParam*/)
{
HtmlHelpW (NULL, HTML_HELP_FILE, HH_DISPLAY_TOPIC, 0);
return TRUE;
}