1890 lines
48 KiB
C++
1890 lines
48 KiB
C++
|
/*++
|
||
|
|
||
|
Copyright (C) 1996-1999 Microsoft Corporation
|
||
|
|
||
|
Module Name:
|
||
|
|
||
|
ctrprop.cpp
|
||
|
|
||
|
Abstract:
|
||
|
|
||
|
This file contains the CCounterPropPage class and other routines
|
||
|
to implement the counter property page.
|
||
|
|
||
|
--*/
|
||
|
|
||
|
#include <assert.h>
|
||
|
#include <stdio.h>
|
||
|
#include <math.h>
|
||
|
#include <limits.h>
|
||
|
#include "polyline.h"
|
||
|
#include "ctrprop.h"
|
||
|
#include "utils.h"
|
||
|
#include "unihelpr.h"
|
||
|
#include "winhelpr.h"
|
||
|
#include "visuals.h"
|
||
|
#include "strids.h"
|
||
|
#include "winperf.h"
|
||
|
#include "pdhmsg.h"
|
||
|
#include "globals.h"
|
||
|
#include "browser.h"
|
||
|
#include "smonmsg.h"
|
||
|
|
||
|
#define OWNER_DRAWN_ITEM 2
|
||
|
#define OWNER_DRAW_FOCUS 1
|
||
|
|
||
|
|
||
|
VOID static
|
||
|
HandleSelectionState (
|
||
|
IN LPDRAWITEMSTRUCT lpdis
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
HandleSelectionState draws or erases a selection rectangle around an item
|
||
|
in a combo box list.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
lpdis - Pointer to DRAWITEMSTRUCT
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
HBRUSH hbr ;
|
||
|
|
||
|
if ( NULL != lpdis ) {
|
||
|
|
||
|
if (lpdis->itemState & ODS_SELECTED)
|
||
|
hbr = (HBRUSH)GetStockObject(BLACK_BRUSH) ;
|
||
|
else
|
||
|
hbr = CreateSolidBrush(GetSysColor(COLOR_WINDOW)) ;
|
||
|
|
||
|
if ( NULL != hbr ) {
|
||
|
FrameRect(lpdis->hDC, (LPRECT)&lpdis->rcItem, hbr) ;
|
||
|
DeleteObject (hbr) ;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
//***************************************************************************
|
||
|
// *
|
||
|
// FUNCTION : HandleFocusState(LPDRAWITEMSTRUCT) *
|
||
|
// *
|
||
|
// PURPOSE : Handle a change in item focus state. If an item gains the *
|
||
|
// input focus, a gray rectangular frame is drawn around that *
|
||
|
// item; if an item loses the input focus, the gray frame is *
|
||
|
// removed. *
|
||
|
// *
|
||
|
// COMMENT : The gray focus frame is slightly smaller than the black *
|
||
|
// selection frame so they won't paint over each other. *
|
||
|
// *
|
||
|
//***************************************************************************
|
||
|
VOID static
|
||
|
HandleFocusState (
|
||
|
IN LPDRAWITEMSTRUCT lpdis
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
HandleFocusState draws or erases a focus rectangle around an item in
|
||
|
the pulldown list of a combo box. The reactngle is indented to not
|
||
|
interfere with the selection rectangle.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
lpdis - Pointer to DRAWITEMSTRUCT
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
RECT rc ;
|
||
|
HBRUSH hbr ;
|
||
|
|
||
|
if ( NULL != lpdis ) {
|
||
|
|
||
|
// Resize rectangle to place focus frame between the selection
|
||
|
// frame and the item.
|
||
|
CopyRect ((LPRECT)&rc, (LPRECT)&lpdis->rcItem) ;
|
||
|
InflateRect ((LPRECT)&rc, -OWNER_DRAW_FOCUS, -OWNER_DRAW_FOCUS) ;
|
||
|
|
||
|
// Gray if has focus, background color if not
|
||
|
if (lpdis->itemState & ODS_FOCUS)
|
||
|
hbr = (HBRUSH)GetStockObject(GRAY_BRUSH) ;
|
||
|
else
|
||
|
hbr = CreateSolidBrush(GetSysColor(COLOR_WINDOW)) ;
|
||
|
|
||
|
if ( NULL != hbr ) {
|
||
|
FrameRect(lpdis->hDC, (LPRECT)&rc, hbr) ;
|
||
|
DeleteObject (hbr) ;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
CCounterPropPage::CCounterPropPage ( void )
|
||
|
: m_pInfoSel ( NULL ),
|
||
|
m_pInfoDeleted ( NULL ),
|
||
|
m_dwMaxHorizListExtent ( 0 ),
|
||
|
m_nLastSelCount( 0 ),
|
||
|
m_bAreModSelectedVisuals ( FALSE ),
|
||
|
m_fHashTableSetup ( FALSE )
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
Creation routine for counter property page. Initializes the instance
|
||
|
variables.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
None.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
m_uIDDialog = IDD_CTR_PROPP_DLG;
|
||
|
m_uIDTitle = IDS_CTR_PROPP_TITLE;
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
CCounterPropPage::~CCounterPropPage (
|
||
|
VOID
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
Destructor for counter property page.
|
||
|
Arguments:
|
||
|
|
||
|
None.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
ClearCountersHashTable();
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
CCounterPropPage::DeinitControls ( void )
|
||
|
{
|
||
|
ISystemMonitor *pObj = NULL;
|
||
|
CImpISystemMonitor *pPrivObj = NULL;
|
||
|
HWND hwndList = NULL;
|
||
|
INT iIndex = 0;
|
||
|
INT iItemCnt = 0;
|
||
|
PItemInfo pInfo = NULL;
|
||
|
|
||
|
// Write the current visuals back to the control
|
||
|
// Must be at least one control object and only the first is used
|
||
|
if (m_cObjects != 0) {
|
||
|
pObj = m_ppISysmon[0];
|
||
|
if ( NULL != pObj ) {
|
||
|
pPrivObj = (CImpISystemMonitor*)pObj;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( NULL != pPrivObj ) {
|
||
|
if ( AreModifiedSelectedVisuals() ) {
|
||
|
IncrementLocalVisuals();
|
||
|
}
|
||
|
pPrivObj->SetVisuals(m_props.rgbColor, m_props.iColorIndex, m_props.iWidthIndex, m_props.iStyleIndex);
|
||
|
}
|
||
|
|
||
|
hwndList = DialogControl(m_hDlg, IDC_CTRLIST);
|
||
|
if ( NULL != hwndList ) {
|
||
|
iItemCnt = LBNumItems(hwndList);
|
||
|
for (iIndex = 0; iIndex < iItemCnt; iIndex++ ) {
|
||
|
pInfo = (PItemInfo)LBData(hwndList,iIndex);
|
||
|
if ( NULL != pInfo ) {
|
||
|
DeleteInfo(pInfo);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
INT
|
||
|
CCounterPropPage::SelectMatchingItem (
|
||
|
INT iColorIndex,
|
||
|
COLORREF rgbCustomColor,
|
||
|
INT iWidthIndex,
|
||
|
INT iStyleIndex)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
GetMatchingIndex selects the first counter item that matches the
|
||
|
specified visual characteristics.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
None.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
Returns the index of the selected item on match. If none match, then
|
||
|
returns -1.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
INT iReturn = -1;
|
||
|
|
||
|
// Get number of items in list box
|
||
|
HWND hwndList = DialogControl(m_hDlg, IDC_CTRLIST);
|
||
|
INT iItemCnt = LBNumItems(hwndList);
|
||
|
INT i;
|
||
|
bool bMatch = false;
|
||
|
|
||
|
// For each item
|
||
|
for (i=0; !bMatch && i<iItemCnt; i++){
|
||
|
|
||
|
PItemInfo pInfo = (PItemInfo)LBData(hwndList,i);
|
||
|
|
||
|
LoadItemProps ( pInfo );
|
||
|
|
||
|
if ( pInfo->Props.iColorIndex == iColorIndex ) {
|
||
|
if ( NumStandardColorIndices() > iColorIndex ) {
|
||
|
bMatch = true;
|
||
|
} else {
|
||
|
if ( pInfo->Props.rgbColor == rgbCustomColor )
|
||
|
bMatch = true;
|
||
|
}
|
||
|
|
||
|
if ( bMatch ) {
|
||
|
if ( ( iWidthIndex != pInfo->Props.iWidthIndex )
|
||
|
|| ( iStyleIndex != pInfo->Props.iStyleIndex ) ){
|
||
|
bMatch = false;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( bMatch ) {
|
||
|
iReturn = i - 1;
|
||
|
|
||
|
SelectItem ( iReturn );
|
||
|
}
|
||
|
|
||
|
return iReturn;
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL
|
||
|
CCounterPropPage::GetProperties (
|
||
|
VOID
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
GetProperties initializes the dialog box for the property page. It then
|
||
|
fetches an ICounterItem interface for each counter of the control being
|
||
|
edited. Each interface pointer is placed in an ItemInfo structure which is
|
||
|
then added to the dialog list box.
|
||
|
|
||
|
The counter properties are not fetched until they are needed for display.
|
||
|
The first counter fetched is selected to display its properties in the
|
||
|
dialog box.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
None.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
Boolean status - TRUE = success
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
ISystemMonitor *pObj = NULL;
|
||
|
CImpISystemMonitor *pPrivObj = NULL;
|
||
|
ICounterItem *pItem;
|
||
|
ICounterItem *pSelectedItem = NULL;
|
||
|
PItemInfo pInfo;
|
||
|
INT nCtr;
|
||
|
INT nSelCtr = LB_ERR;
|
||
|
INT iIndex;
|
||
|
INT nChar;
|
||
|
BOOL bStat = TRUE;
|
||
|
BSTR bstrPath;
|
||
|
HRESULT dwResult;
|
||
|
PPDH_COUNTER_PATH_ELEMENTS pCounter;
|
||
|
|
||
|
InitDialog();
|
||
|
|
||
|
// Must be at least one control object and only the first is used
|
||
|
if (m_cObjects == 0) {
|
||
|
bStat = FALSE;
|
||
|
} else {
|
||
|
pObj = m_ppISysmon[0];
|
||
|
if ( NULL != pObj ) {
|
||
|
pPrivObj = (CImpISystemMonitor*)pObj;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( bStat && NULL != pObj && NULL != pPrivObj ) {
|
||
|
|
||
|
// Request each counter from the control, stopping on failure
|
||
|
nCtr = 0;
|
||
|
|
||
|
pPrivObj->GetSelectedCounter( &pSelectedItem );
|
||
|
|
||
|
while (SUCCEEDED(pObj->Counter(nCtr, &pItem))) {
|
||
|
// Create ItemInfo to hold the counter
|
||
|
pInfo = new ItemInfo;
|
||
|
|
||
|
if (pInfo == NULL) {
|
||
|
bStat = FALSE;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
ZeroMemory(pInfo, sizeof(ItemInfo));
|
||
|
|
||
|
pInfo->pItem = pItem;
|
||
|
pItem->get_Path( &bstrPath );
|
||
|
|
||
|
if (bstrPath != NULL) {
|
||
|
nChar = lstrlen(bstrPath) + 1;
|
||
|
|
||
|
pInfo->pszPath = new WCHAR [nChar];
|
||
|
|
||
|
if (pInfo->pszPath == NULL) {
|
||
|
delete pInfo;
|
||
|
pInfo = NULL;
|
||
|
SysFreeString(bstrPath);
|
||
|
bStat = FALSE;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
StringCchCopy(pInfo->pszPath, nChar, bstrPath);
|
||
|
|
||
|
SysFreeString(bstrPath);
|
||
|
}
|
||
|
|
||
|
dwResult = InsertCounterToHashTable(pInfo->pszPath, &pCounter);
|
||
|
|
||
|
if (dwResult == ERROR_SUCCESS) {
|
||
|
//
|
||
|
// Add the counter to the list box
|
||
|
//
|
||
|
iIndex = AddItemToList(pInfo);
|
||
|
|
||
|
if ( LB_ERR == iIndex ) {
|
||
|
RemoveCounterFromHashTable(pInfo->pszPath, pCounter);
|
||
|
bStat = FALSE;
|
||
|
DeleteInfo(pInfo);
|
||
|
pInfo = NULL;
|
||
|
break;
|
||
|
} else {
|
||
|
pInfo->pCounter = pCounter;
|
||
|
if ( pSelectedItem == pItem ) {
|
||
|
nSelCtr = iIndex;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
bStat = FALSE;
|
||
|
DeleteInfo(pInfo);
|
||
|
pInfo = NULL;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
nCtr++;
|
||
|
}
|
||
|
|
||
|
if ( NULL != pSelectedItem ) {
|
||
|
pSelectedItem->Release();
|
||
|
}
|
||
|
|
||
|
// Get the current visuals fron the control
|
||
|
// and initialize the property dialog
|
||
|
pPrivObj->GetVisuals(&m_props.rgbColor, &m_props.iColorIndex, &m_props.iWidthIndex, &m_props.iStyleIndex);
|
||
|
|
||
|
// If a counter matches the selected counter, select that item.
|
||
|
// Else if the visuals match an existing item, select that item.
|
||
|
// Else if there is at least one counter in the control, select the first counter.
|
||
|
// Otherwise, set the display properties to the first counter to be added.
|
||
|
|
||
|
if ( LB_ERR != nSelCtr ) {
|
||
|
SelectItem ( nSelCtr );
|
||
|
} else {
|
||
|
if ( LB_ERR == SelectMatchingItem (
|
||
|
m_props.iColorIndex,
|
||
|
m_props.rgbColor,
|
||
|
m_props.iWidthIndex,
|
||
|
m_props.iStyleIndex ) )
|
||
|
{
|
||
|
if ( 0 < nCtr ) {
|
||
|
SelectItem ( 0 );
|
||
|
} else {
|
||
|
// Init the scale factor to the default
|
||
|
m_props.iScaleIndex = 0;
|
||
|
|
||
|
// If nothing selected, ensure that the color index is set to
|
||
|
// a standard color.
|
||
|
if ( m_props.iColorIndex == NumStandardColorIndices() ) {
|
||
|
m_props.iColorIndex -= 1;
|
||
|
}
|
||
|
|
||
|
CBSetSelection(DialogControl(m_hDlg, IDC_LINECOLOR), m_props.iColorIndex);
|
||
|
CBSetSelection(DialogControl(m_hDlg, IDC_LINEWIDTH), m_props.iWidthIndex);
|
||
|
CBSetSelection(DialogControl(m_hDlg, IDC_LINESTYLE), m_props.iStyleIndex);
|
||
|
CBSetSelection(DialogControl(m_hDlg, IDC_LINESCALE), m_props.iScaleIndex);
|
||
|
SetStyleComboEnable();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return bStat;
|
||
|
}
|
||
|
|
||
|
INT
|
||
|
CCounterPropPage::AddItemToList (
|
||
|
IN PItemInfo pInfo
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
AddItemToList adds a counter's path name to the dialog list box and
|
||
|
attaches a pointer to the counter's ItemInfo structure as item data.
|
||
|
It also adjusts the horizontal scroll of the list box.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
pInfo - Pointer to counter's ItemInfo structure
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
List box index of added counter (LB_ERR on failure)
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
INT iIndex;
|
||
|
HWND hwndList = DialogControl(m_hDlg, IDC_CTRLIST);
|
||
|
DWORD dwItemExtent = 0;
|
||
|
HDC hDC = NULL;
|
||
|
|
||
|
iIndex = (INT)LBAdd(hwndList, pInfo->pszPath);
|
||
|
|
||
|
if ( LB_ERR != iIndex && LB_ERRSPACE != iIndex ) {
|
||
|
LBSetData(hwndList, iIndex, pInfo);
|
||
|
|
||
|
hDC = GetDC ( hwndList );
|
||
|
if ( NULL != hDC ) {
|
||
|
dwItemExtent = (DWORD)TextWidth ( hDC, pInfo->pszPath );
|
||
|
|
||
|
if (dwItemExtent > m_dwMaxHorizListExtent) {
|
||
|
m_dwMaxHorizListExtent = dwItemExtent;
|
||
|
LBSetHorzExtent ( hwndList, dwItemExtent );
|
||
|
}
|
||
|
ReleaseDC (hwndList, hDC) ;
|
||
|
}
|
||
|
} else {
|
||
|
iIndex = LB_ERR ;
|
||
|
}
|
||
|
return iIndex;
|
||
|
}
|
||
|
|
||
|
VOID
|
||
|
CCounterPropPage::LoadItemProps (
|
||
|
IN PItemInfo pInfo
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
LoadItemProps loads the properties of the selected counter through the
|
||
|
counter's interface into the ItemInfo structure, if not already loaded.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
pInfo - pointer to item info
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
// If properties not loaded for this item, get them now
|
||
|
if (pInfo->pItem && !pInfo->fLoaded) {
|
||
|
INT iScaleFactor;
|
||
|
INT iStyle;
|
||
|
INT iWidth;
|
||
|
pInfo->pItem->get_Color ( &pInfo->Props.rgbColor );
|
||
|
pInfo->pItem->get_ScaleFactor ( &iScaleFactor );
|
||
|
pInfo->pItem->get_Width ( &iWidth );
|
||
|
pInfo->pItem->get_LineStyle ( &iStyle );
|
||
|
// Translate to combo box indices
|
||
|
pInfo->Props.iColorIndex = ColorToIndex ( pInfo->Props.rgbColor );
|
||
|
pInfo->Props.iStyleIndex = StyleToIndex ( iStyle );
|
||
|
pInfo->Props.iWidthIndex = WidthToIndex ( iWidth );
|
||
|
pInfo->Props.iScaleIndex = ScaleFactorToIndex ( iScaleFactor );
|
||
|
|
||
|
pInfo->fLoaded = TRUE;
|
||
|
}
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
VOID
|
||
|
CCounterPropPage::DisplayItemProps (
|
||
|
IN PItemInfo pInfo
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
DisplayItemProps displays the properties of the selected counter on the
|
||
|
property page dialog. If the counter is being displayed for the first time
|
||
|
the properties are obtained through the counter's interface and are loaded
|
||
|
into the ItemInfo structure.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
pInfo - pointer to item info
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
// Get number of items in color combo box
|
||
|
HWND hWndColor = DialogControl(m_hDlg, IDC_LINECOLOR);
|
||
|
INT iCurrentColorCnt = CBNumItems(hWndColor);
|
||
|
|
||
|
// If properties not loaded for this item, get them now
|
||
|
LoadItemProps ( pInfo );
|
||
|
|
||
|
// Display the properties
|
||
|
m_props = pInfo->Props;
|
||
|
|
||
|
// Handle custom color
|
||
|
if ( iCurrentColorCnt > NumStandardColorIndices() ) {
|
||
|
// Delete the custom color item. It is stored at
|
||
|
// the end of the list.
|
||
|
CBDelete(hWndColor, iCurrentColorCnt - 1);
|
||
|
}
|
||
|
|
||
|
// If new custom color, add it at the end of the list.
|
||
|
if ( NumStandardColorIndices() == m_props.iColorIndex )
|
||
|
CBAdd( hWndColor, (INT_PTR)m_props.iColorIndex );
|
||
|
|
||
|
CBSetSelection(hWndColor, m_props.iColorIndex);
|
||
|
CBSetSelection(DialogControl(m_hDlg, IDC_LINEWIDTH), m_props.iWidthIndex);
|
||
|
CBSetSelection(DialogControl(m_hDlg, IDC_LINESTYLE), m_props.iStyleIndex);
|
||
|
CBSetSelection(DialogControl(m_hDlg, IDC_LINESCALE), m_props.iScaleIndex);
|
||
|
|
||
|
SetStyleComboEnable();
|
||
|
}
|
||
|
|
||
|
|
||
|
BOOL
|
||
|
CCounterPropPage::SetProperties (
|
||
|
VOID
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
SetProperties applies the counter changes the user has made. It calls the
|
||
|
control's AddCounter and DeleteCounter to adjust the counter set. It calls
|
||
|
the counter's property functions for all new and changed counters.
|
||
|
|
||
|
The counters to be deleted are in the pInfoDeleted linked list. The other
|
||
|
counters are obtained from the dialog list box.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
None.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
Boolean status - TRUE = success
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
HWND hwndList;
|
||
|
INT iItemCnt;
|
||
|
INT i;
|
||
|
PItemInfo pInfo, pInfoNext;
|
||
|
ISystemMonitor *pObj;
|
||
|
BSTR pBstr = NULL;
|
||
|
|
||
|
// Apply changes to first control
|
||
|
pObj = m_ppISysmon[0];
|
||
|
|
||
|
if ( NULL != pObj ) {
|
||
|
|
||
|
// For all items in the delete list
|
||
|
pInfo = m_pInfoDeleted;
|
||
|
while (pInfo) {
|
||
|
|
||
|
// If this counter exists in the control
|
||
|
if (pInfo->pItem != NULL) {
|
||
|
|
||
|
// Tell control to remove it
|
||
|
pObj->DeleteCounter(pInfo->pItem);
|
||
|
}
|
||
|
|
||
|
// Delete the Info structure and point to the next one
|
||
|
pInfoNext = pInfo->pNextInfo;
|
||
|
|
||
|
DeleteInfo(pInfo);
|
||
|
|
||
|
pInfo = pInfoNext;
|
||
|
}
|
||
|
|
||
|
m_pInfoDeleted = NULL;
|
||
|
|
||
|
// Get number of items in list box
|
||
|
hwndList = DialogControl(m_hDlg, IDC_CTRLIST);
|
||
|
iItemCnt = LBNumItems(hwndList);
|
||
|
|
||
|
//assert( IsWindowUnicode(hwndList) );
|
||
|
|
||
|
// For each item
|
||
|
for (i=0; i<iItemCnt; i++) {
|
||
|
pInfo = (PItemInfo)LBData(hwndList,i);
|
||
|
|
||
|
// If new item, create it now
|
||
|
if (pInfo->pItem == NULL) {
|
||
|
#if UNICODE
|
||
|
pBstr = SysAllocString(pInfo->pszPath);
|
||
|
pObj->AddCounter(pBstr, &pInfo->pItem);
|
||
|
#else
|
||
|
INT nChar = lstrlen(pInfo->pszPath);
|
||
|
LPWSTR pszPathW = new WCHAR [nChar + 1];
|
||
|
if (pszPathW) {
|
||
|
MultiByteToWideChar(CP_ACP, 0, pInfo->pszPath, nChar+1, pszPathW, nChar+1);
|
||
|
pBstr = SysAllocString(pInfo->pszPath);
|
||
|
pObj->AddCounter(pBstr, &pInfo->pItem);
|
||
|
delete [] pszPathW;
|
||
|
}
|
||
|
#endif
|
||
|
if (pBstr) {
|
||
|
SysFreeString(pBstr);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// If item has changed, put the new properties
|
||
|
if (pInfo->pItem != NULL && pInfo->fChanged) {
|
||
|
// iColorIndex is used to determine standard colors.
|
||
|
if ( pInfo->Props.iColorIndex < NumStandardColorIndices() ) {
|
||
|
pInfo->pItem->put_Color(IndexToStandardColor( pInfo->Props.iColorIndex) );
|
||
|
} else {
|
||
|
pInfo->pItem->put_Color(pInfo->Props.rgbColor);
|
||
|
}
|
||
|
|
||
|
pInfo->pItem->put_Width(IndexToWidth(pInfo->Props.iWidthIndex));
|
||
|
pInfo->pItem->put_LineStyle(IndexToStyle(pInfo->Props.iStyleIndex));
|
||
|
pInfo->pItem->put_ScaleFactor(IndexToScaleFactor( pInfo->Props.iScaleIndex ) );
|
||
|
|
||
|
pInfo->fChanged = FALSE;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Tell graph to redraw itself
|
||
|
pObj->UpdateGraph();
|
||
|
} // else report internal error
|
||
|
return TRUE;
|
||
|
}
|
||
|
|
||
|
|
||
|
VOID
|
||
|
CCounterPropPage::InitDialog (
|
||
|
VOID
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
InitDialog loads each attribute combo box with its list of choices and
|
||
|
selects the default choice. The graphical attributes are owner drawn, so
|
||
|
their list items are just set to numerical indices. The scale attribute
|
||
|
list is filled with numeric strings representing scale factors plus a
|
||
|
default selection.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
None.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
HWND hWndColors;
|
||
|
HWND hWndWidths;
|
||
|
HWND hWndStyles;
|
||
|
HWND hWndScales;
|
||
|
INT i ;
|
||
|
double ScaleFactor ;
|
||
|
WCHAR tempBuff[MAX_VALUE_LEN] ;
|
||
|
|
||
|
|
||
|
if (m_hDlg == NULL)
|
||
|
return;
|
||
|
//assert( IsWindowUnicode( m_hDlg ) );
|
||
|
|
||
|
// Load the colors combobox, select the default color.
|
||
|
hWndColors = DialogControl (m_hDlg, IDC_LINECOLOR) ;
|
||
|
for (i = 0 ; i < NumStandardColorIndices () ; i++)
|
||
|
CBAdd (hWndColors, (INT_PTR)1); // string pointer is unused. Fill with
|
||
|
// arbitrary value.
|
||
|
|
||
|
CBSetSelection (hWndColors, 0) ;
|
||
|
|
||
|
// Load the widths combo box, select the default width.
|
||
|
hWndWidths = DialogControl (m_hDlg, IDC_LINEWIDTH) ;
|
||
|
for (i = 0 ; i < NumWidthIndices () ; i++)
|
||
|
CBAdd (hWndWidths, (INT_PTR)1) ;
|
||
|
|
||
|
CBSetSelection (hWndWidths, 0) ;
|
||
|
|
||
|
// Load the styles combo box, select the default style.
|
||
|
hWndStyles = DialogControl (m_hDlg, IDC_LINESTYLE) ;
|
||
|
for (i = 0 ; i < NumStyleIndices () ; i++)
|
||
|
CBAdd (hWndStyles, (INT_PTR)1) ;
|
||
|
|
||
|
CBSetSelection (hWndStyles, 0) ;
|
||
|
|
||
|
// Init the scale combo box.
|
||
|
hWndScales = DialogControl (m_hDlg, IDC_LINESCALE) ;
|
||
|
|
||
|
CBAdd (hWndScales, ResourceString(IDS_DEFAULT)) ;
|
||
|
|
||
|
// Generate power of 10 scale factors
|
||
|
ScaleFactor = pow (10.0f, (double)PDH_MIN_SCALE);
|
||
|
for (i = PDH_MIN_SCALE ; i <= PDH_MAX_SCALE ; i++) {
|
||
|
|
||
|
FormatNumber (
|
||
|
ScaleFactor,
|
||
|
tempBuff,
|
||
|
MAX_VALUE_LEN,
|
||
|
(i <= 0 ? (-1 * i) + 1 : i + 1),
|
||
|
(i < 0 ? (-1 * i) : 1) );
|
||
|
|
||
|
CBAdd (hWndScales, tempBuff) ;
|
||
|
|
||
|
ScaleFactor *= (double) 10.0f ;
|
||
|
}
|
||
|
|
||
|
CBSetSelection (hWndScales, 0) ;
|
||
|
ClearCountersHashTable();
|
||
|
|
||
|
DialogEnable(m_hDlg, IDC_LINECOLOR, FALSE);
|
||
|
DialogEnable(m_hDlg, IDC_LINEWIDTH, FALSE);
|
||
|
DialogEnable(m_hDlg, IDC_LINESTYLE, FALSE);
|
||
|
DialogEnable(m_hDlg, IDC_LINESCALE, FALSE);
|
||
|
}
|
||
|
|
||
|
|
||
|
void
|
||
|
CCounterPropPage::IncrementLocalVisuals (
|
||
|
void
|
||
|
)
|
||
|
{
|
||
|
// Increment the visual indices in color, width, style order
|
||
|
if (++m_props.iColorIndex >= NumStandardColorIndices()) {
|
||
|
m_props.iColorIndex = 0;
|
||
|
|
||
|
if (++m_props.iWidthIndex >= NumWidthIndices()) {
|
||
|
m_props.iWidthIndex = 0;
|
||
|
|
||
|
if (++m_props.iStyleIndex < NumStyleIndices()) {
|
||
|
m_props.iStyleIndex = 0;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
SetModifiedSelectedVisuals ( FALSE );
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
HRESULT
|
||
|
CCounterPropPage::NewItem (
|
||
|
IN LPWSTR pszPath,
|
||
|
IN DWORD /* dwFlags */
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
NewItem adds a new counter to the dialog's counter list box. It first
|
||
|
creates a new ItemInfo structure and loads it with the counter pathname
|
||
|
string. Then the ItemInfo is added to the dialog list box.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
pszPath - Pointer to counter pathname string
|
||
|
fGenerated - TRUE if path was generated from a wildcard path
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
Index of new item in counter list (-1 if failed to add)
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
PItemInfo pInfo;
|
||
|
HRESULT dwResult;
|
||
|
BOOL bRet;
|
||
|
PPDH_COUNTER_PATH_ELEMENTS pCounter;
|
||
|
|
||
|
dwResult = InsertCounterToHashTable(pszPath, &pCounter);
|
||
|
|
||
|
if (dwResult != ERROR_SUCCESS) {
|
||
|
return dwResult;
|
||
|
}
|
||
|
|
||
|
// Allocate ItemInfo structure
|
||
|
pInfo = NULL;
|
||
|
pInfo = new ItemInfo;
|
||
|
if (pInfo == NULL) {
|
||
|
bRet = RemoveCounterFromHashTable(pszPath, pCounter);
|
||
|
|
||
|
assert(bRet);
|
||
|
return E_OUTOFMEMORY;
|
||
|
}
|
||
|
|
||
|
// Mark as loaded to prevent requesting attributes from control
|
||
|
// Mark as changed so sttribute will be written
|
||
|
pInfo->fLoaded = TRUE;
|
||
|
pInfo->fChanged = TRUE;
|
||
|
|
||
|
// Actual counter doesn't exist yet
|
||
|
pInfo->pItem = NULL;
|
||
|
|
||
|
// If a counter is selected, we're showing its visuals
|
||
|
// so increment them for the new counter
|
||
|
if (m_pInfoSel != NULL) {
|
||
|
IncrementLocalVisuals();
|
||
|
}
|
||
|
else {
|
||
|
// Point to the new item so the visuals are incremented
|
||
|
// for the next one
|
||
|
m_pInfoSel = pInfo;
|
||
|
}
|
||
|
|
||
|
// Set default scaling
|
||
|
m_props.iScaleIndex = 0;
|
||
|
|
||
|
// Color is non-standard only if user is able to build a color.
|
||
|
if( m_props.iColorIndex < NumStandardColorIndices() )
|
||
|
m_props.rgbColor = IndexToStandardColor( m_props.iColorIndex );
|
||
|
else
|
||
|
m_props.rgbColor = pInfo->Props.rgbColor;
|
||
|
|
||
|
// Copy properties to new counter
|
||
|
pInfo->Props = m_props;
|
||
|
|
||
|
// Make own copy of path name string
|
||
|
pInfo->pszPath = new WCHAR [lstrlen(pszPath) + 1];
|
||
|
|
||
|
if (pInfo->pszPath == NULL)
|
||
|
{
|
||
|
bRet = RemoveCounterFromHashTable(pszPath, pCounter);
|
||
|
|
||
|
assert(bRet);
|
||
|
|
||
|
delete pInfo;
|
||
|
return E_OUTOFMEMORY;
|
||
|
}
|
||
|
|
||
|
StringCchCopy(pInfo->pszPath, lstrlen(pszPath) + 1, pszPath);
|
||
|
|
||
|
// Add to dialog's counter list
|
||
|
pInfo->pCounter = pCounter;
|
||
|
m_iAddIndex = AddItemToList(pInfo);
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
|
||
|
VOID
|
||
|
CCounterPropPage::SelectItem (
|
||
|
IN INT iItem
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
SelectItem selects a specified counter item in the dialog counter list.
|
||
|
It then displays the selected counter's properties and enables the
|
||
|
"Delete Counter" button.
|
||
|
|
||
|
SelectItem can be called with a -1 to deselect all counters and disable
|
||
|
the delete button.
|
||
|
|
||
|
The member variable, m_pInfoSel, is updated to point to the selected
|
||
|
counter info.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
iItem - List index of counter item to select, or -1 to deselect all
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
HWND hWnd;
|
||
|
|
||
|
hWnd = DialogControl(m_hDlg, IDC_CTRLIST);
|
||
|
|
||
|
// Translate index into item pointer
|
||
|
if (iItem == -1) {
|
||
|
m_pInfoSel = NULL;
|
||
|
}
|
||
|
else {
|
||
|
m_pInfoSel = (PItemInfo)LBData(hWnd, iItem);
|
||
|
|
||
|
if ((INT_PTR)m_pInfoSel == LB_ERR)
|
||
|
m_pInfoSel = NULL;
|
||
|
}
|
||
|
|
||
|
// Select the item, display properties, and enable delete button
|
||
|
if (m_pInfoSel != NULL) {
|
||
|
INT nSelCount;
|
||
|
|
||
|
LBSetSel(hWnd, TRUE, iItem);
|
||
|
nSelCount = LBSelectCount(hWnd);
|
||
|
|
||
|
if (nSelCount == 1) {
|
||
|
DialogEnable(m_hDlg, IDC_LINECOLOR, TRUE);
|
||
|
DialogEnable(m_hDlg, IDC_LINEWIDTH, TRUE);
|
||
|
DialogEnable(m_hDlg, IDC_LINESTYLE, TRUE);
|
||
|
DialogEnable(m_hDlg, IDC_LINESCALE, TRUE);
|
||
|
}
|
||
|
else {
|
||
|
DialogEnable(m_hDlg, IDC_LINECOLOR, FALSE);
|
||
|
DialogEnable(m_hDlg, IDC_LINEWIDTH, FALSE);
|
||
|
DialogEnable(m_hDlg, IDC_LINESTYLE, FALSE);
|
||
|
DialogEnable(m_hDlg, IDC_LINESCALE, FALSE);
|
||
|
}
|
||
|
m_nLastSelCount = nSelCount;
|
||
|
DisplayItemProps(m_pInfoSel);
|
||
|
DialogEnable(m_hDlg,IDC_DELCTR,1);
|
||
|
}
|
||
|
else {
|
||
|
LBSetSel(hWnd, FALSE, -1);
|
||
|
DialogEnable(m_hDlg,IDC_DELCTR, 0);
|
||
|
DialogEnable(m_hDlg, IDC_LINECOLOR, FALSE);
|
||
|
DialogEnable(m_hDlg, IDC_LINEWIDTH, FALSE);
|
||
|
DialogEnable(m_hDlg, IDC_LINESTYLE, FALSE);
|
||
|
DialogEnable(m_hDlg, IDC_LINESCALE, FALSE);
|
||
|
m_nLastSelCount = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
VOID
|
||
|
CCounterPropPage::DeleteItem (
|
||
|
VOID
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
DeleteItem removes the currently selected counter from the dialog's counter
|
||
|
listbox. It adds the item to the deletion list, so the actual counter can
|
||
|
be deleted from the control when (and if) the changes are applied.
|
||
|
|
||
|
The routine selects selects the next counter in the listbox if there is one.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
None.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
HWND hWnd;
|
||
|
INT iIndex;
|
||
|
PItemInfo pInfo;
|
||
|
DWORD dwItemExtent = 0;
|
||
|
HDC hDC = NULL;
|
||
|
INT iStartIndex;
|
||
|
INT iEndIndex;
|
||
|
INT iSelectCount;
|
||
|
INT iItemCount;
|
||
|
INT iOrigCaret;
|
||
|
|
||
|
// Get selected index
|
||
|
hWnd = DialogControl(m_hDlg, IDC_CTRLIST);
|
||
|
iSelectCount = LBSelectCount(hWnd);
|
||
|
if (iSelectCount <= 0 ) {
|
||
|
return;
|
||
|
}
|
||
|
else {
|
||
|
iEndIndex = 0;
|
||
|
iStartIndex = LBNumItems(hWnd) - 1;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// We scan the whole list box to remove items
|
||
|
// May use LB_GETSELITEMS (did not compare the performance)?
|
||
|
//
|
||
|
iOrigCaret = LBFocus(hWnd);
|
||
|
for (iIndex = iStartIndex; iIndex >= iEndIndex; iIndex--) {
|
||
|
if (!LBSelected(hWnd, iIndex)) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// Get selected item info
|
||
|
pInfo = (PItemInfo)LBData(hWnd, iIndex);
|
||
|
|
||
|
// Move it to the "Deleted" list.
|
||
|
pInfo->pNextInfo = m_pInfoDeleted;
|
||
|
m_pInfoDeleted = pInfo;
|
||
|
|
||
|
// Remove the string from the list box.
|
||
|
LBDelete(hWnd, iIndex);
|
||
|
|
||
|
// Remove the counter from hash table
|
||
|
RemoveCounterFromHashTable(pInfo->pszPath, pInfo->pCounter);
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// see how many entries are left and update the
|
||
|
// caret position and the remove button state
|
||
|
//
|
||
|
iItemCount = LBNumItems(hWnd);
|
||
|
if (iItemCount > 0) {
|
||
|
// the update the caret
|
||
|
if (iOrigCaret >= iItemCount) {
|
||
|
iOrigCaret = iItemCount-1;
|
||
|
} else {
|
||
|
// caret should be within the list
|
||
|
}
|
||
|
SelectItem(iOrigCaret);
|
||
|
} else {
|
||
|
// the list is empty so remove caret, selection
|
||
|
// disable the remove button and activate the
|
||
|
// add button
|
||
|
SelectItem(-1);
|
||
|
}
|
||
|
|
||
|
// Clear the max horizontal extent and recalculate
|
||
|
m_dwMaxHorizListExtent = 0;
|
||
|
|
||
|
hDC = GetDC ( hWnd );
|
||
|
if ( NULL != hDC ) {
|
||
|
for ( iIndex = 0; iIndex < (INT)LBNumItems ( hWnd ); iIndex++ ) {
|
||
|
pInfo = (PItemInfo)LBData(hWnd, iIndex);
|
||
|
dwItemExtent = (DWORD)TextWidth ( hDC, pInfo->pszPath );
|
||
|
if (dwItemExtent > m_dwMaxHorizListExtent) {
|
||
|
m_dwMaxHorizListExtent = dwItemExtent;
|
||
|
}
|
||
|
}
|
||
|
ReleaseDC (hWnd, hDC) ;
|
||
|
}
|
||
|
|
||
|
LBSetHorzExtent ( hWnd, m_dwMaxHorizListExtent );
|
||
|
|
||
|
// Set change flag to enable "Apply" button
|
||
|
SetChange();
|
||
|
}
|
||
|
|
||
|
|
||
|
static HRESULT
|
||
|
AddCallback (
|
||
|
LPWSTR pszPathName,
|
||
|
DWORD_PTR lpUserData,
|
||
|
DWORD dwFlags
|
||
|
)
|
||
|
{
|
||
|
CCounterPropPage* pObj = (CCounterPropPage*)lpUserData;
|
||
|
|
||
|
return pObj->NewItem(pszPathName, dwFlags);
|
||
|
}
|
||
|
|
||
|
|
||
|
VOID
|
||
|
CCounterPropPage::AddCounters (
|
||
|
VOID
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
AddCounters invokes the counter browser to select new counters.
|
||
|
The browser calls the AddCallback function for each new counter.
|
||
|
AddCallback passes the counter path on to the NewItem method.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
None.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
HRESULT hr = NOERROR;
|
||
|
HLOG hDataSource;
|
||
|
VARIANT_BOOL bMonitorDuplicateInstances = FALSE;
|
||
|
ISystemMonitor *pObj = m_ppISysmon[0];
|
||
|
CImpISystemMonitor *pPrivObj;
|
||
|
eDataSourceTypeConstant eDataSource = sysmonNullDataSource;
|
||
|
|
||
|
USES_CONVERSION
|
||
|
|
||
|
pPrivObj = (CImpISystemMonitor*)pObj;
|
||
|
|
||
|
m_iAddIndex = -1;
|
||
|
|
||
|
// Browse counters (calling AddCallack for each selected counter)
|
||
|
hr = pObj->get_MonitorDuplicateInstances(&bMonitorDuplicateInstances);
|
||
|
|
||
|
if (SUCCEEDED(hr)) {
|
||
|
hr = pObj->get_DataSourceType(& eDataSource);
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(hr)) {
|
||
|
// Cannot call pObj->BrowseCounter() because using callback method
|
||
|
// private to this file.
|
||
|
|
||
|
if ( sysmonLogFiles == eDataSource
|
||
|
|| sysmonSqlLog == eDataSource ) {
|
||
|
|
||
|
hDataSource = pPrivObj->GetDataSourceHandle();
|
||
|
assert ( NULL != hDataSource );
|
||
|
} else {
|
||
|
hDataSource = H_REALTIME_DATASOURCE;
|
||
|
}
|
||
|
|
||
|
if (SUCCEEDED(hr)) {
|
||
|
BrowseCounters(
|
||
|
hDataSource,
|
||
|
PERF_DETAIL_WIZARD,
|
||
|
m_hDlg,
|
||
|
AddCallback,
|
||
|
(LPVOID) this,
|
||
|
(BOOL) bMonitorDuplicateInstances);
|
||
|
}
|
||
|
} else {
|
||
|
// Todo: Error message
|
||
|
}
|
||
|
|
||
|
// if any items added, select the last one
|
||
|
if (m_iAddIndex != -1) {
|
||
|
|
||
|
SelectItem(-1);
|
||
|
SelectItem(m_iAddIndex);
|
||
|
m_iAddIndex = -1;
|
||
|
|
||
|
// Set change to enable "Apply" button
|
||
|
SetChange();
|
||
|
}
|
||
|
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
VOID
|
||
|
CCounterPropPage::DialogItemChange (
|
||
|
IN WORD wId,
|
||
|
IN WORD wMsg
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
DialogItemChange processes window messages sent to any of the property
|
||
|
page dialog controls. When the counter listbox selection changes, it
|
||
|
selects the new counter item and displays its properties. When a change is
|
||
|
made to a property combo box, it updates the property for the currently
|
||
|
selected counter item. When the add or delete counter button is pressed,
|
||
|
it calls the appropriate property page functions.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
wID - Dialog control ID
|
||
|
wMsg - Notification code
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None.
|
||
|
|
||
|
--*/
|
||
|
|
||
|
{
|
||
|
INT iIndex;
|
||
|
INT iNewProp;
|
||
|
HWND hWnd;
|
||
|
|
||
|
// Case on control ID
|
||
|
switch (wId) {
|
||
|
|
||
|
case IDC_CTRLIST:
|
||
|
|
||
|
// If selection changed
|
||
|
if (wMsg == LBN_SELCHANGE) {
|
||
|
INT nSelCount;
|
||
|
|
||
|
// Get selected index
|
||
|
hWnd = DialogControl(m_hDlg, IDC_CTRLIST);
|
||
|
|
||
|
nSelCount = LBSelectCount(hWnd);
|
||
|
|
||
|
if (nSelCount == 1) {
|
||
|
if (m_nLastSelCount != 1) {
|
||
|
DialogEnable(m_hDlg, IDC_LINECOLOR, TRUE);
|
||
|
DialogEnable(m_hDlg, IDC_LINEWIDTH, TRUE);
|
||
|
DialogEnable(m_hDlg, IDC_LINESTYLE, TRUE);
|
||
|
DialogEnable(m_hDlg, IDC_LINESCALE, TRUE);
|
||
|
}
|
||
|
|
||
|
if (LBGetSelItems(hWnd, 1, &iIndex) == 1) {
|
||
|
m_pInfoSel = (PItemInfo)LBData(hWnd, iIndex);
|
||
|
DisplayItemProps(m_pInfoSel);
|
||
|
DialogEnable(m_hDlg, IDC_DELCTR, 1);
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
DialogEnable(m_hDlg, IDC_LINECOLOR, FALSE);
|
||
|
DialogEnable(m_hDlg, IDC_LINEWIDTH, FALSE);
|
||
|
DialogEnable(m_hDlg, IDC_LINESTYLE, FALSE);
|
||
|
DialogEnable(m_hDlg, IDC_LINESCALE, FALSE);
|
||
|
}
|
||
|
m_nLastSelCount = nSelCount;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case IDC_LINECOLOR:
|
||
|
case IDC_LINEWIDTH:
|
||
|
case IDC_LINESTYLE:
|
||
|
case IDC_LINESCALE:
|
||
|
|
||
|
// If selection changed and a counter is selected
|
||
|
if (wMsg == CBN_SELCHANGE) {
|
||
|
|
||
|
hWnd = DialogControl(m_hDlg, wId);
|
||
|
iNewProp = (INT)CBSelection(hWnd);
|
||
|
|
||
|
// Store the new property selection
|
||
|
switch (wId) {
|
||
|
|
||
|
case IDC_LINECOLOR:
|
||
|
m_props.iColorIndex = iNewProp;
|
||
|
// If iColorIndex is for the custom color, the
|
||
|
// custom color is already set in the properties.
|
||
|
break;
|
||
|
|
||
|
case IDC_LINEWIDTH:
|
||
|
m_props.iWidthIndex = iNewProp;
|
||
|
SetStyleComboEnable();
|
||
|
break;
|
||
|
|
||
|
case IDC_LINESTYLE:
|
||
|
m_props.iStyleIndex = iNewProp;
|
||
|
break;
|
||
|
|
||
|
case IDC_LINESCALE:
|
||
|
m_props.iScaleIndex = iNewProp;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// If counter is selected, update its properties
|
||
|
if (m_pInfoSel != NULL) {
|
||
|
|
||
|
m_pInfoSel->Props = m_props;
|
||
|
|
||
|
// mark the counter as changed
|
||
|
m_pInfoSel->fChanged = TRUE;
|
||
|
SetChange();
|
||
|
SetModifiedSelectedVisuals( TRUE );
|
||
|
}
|
||
|
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case IDC_ADDCTR:
|
||
|
// Invoke counter browser to add to counter
|
||
|
AddCounters();
|
||
|
break;
|
||
|
|
||
|
case IDC_DELCTR:
|
||
|
// Delete the currently selected counter
|
||
|
DeleteItem();
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
VOID
|
||
|
CCounterPropPage::MeasureItem (
|
||
|
OUT PMEASUREITEMSTRUCT pMI
|
||
|
)
|
||
|
{
|
||
|
pMI->CtlType = ODT_COMBOBOX ;
|
||
|
pMI->CtlID = IDC_LINECOLOR ;
|
||
|
pMI->itemData = 0 ;
|
||
|
pMI->itemWidth = 0 ;
|
||
|
pMI->itemHeight = 14 ;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
VOID
|
||
|
CCounterPropPage::DrawItem (
|
||
|
IN PDRAWITEMSTRUCT pDI
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
DrawItem draws a specified item in one of the graphical property combo
|
||
|
boxes. It is called to process WM_DRAWITEM messages.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
pDI - Pointer to DRAWITEMSTRUCT
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
HDC hDC ;
|
||
|
PRECT prect ;
|
||
|
INT itemID,
|
||
|
CtlID,
|
||
|
itemAction ;
|
||
|
HPEN hPen;
|
||
|
COLORREF rgbBk, rgbOldBk;
|
||
|
|
||
|
if ( NULL != pDI ) {
|
||
|
|
||
|
hDC = pDI->hDC ;
|
||
|
CtlID = pDI->CtlID ;
|
||
|
prect = &pDI->rcItem ;
|
||
|
itemID = pDI->itemID ;
|
||
|
itemAction = pDI->itemAction ;
|
||
|
|
||
|
// Case on drawing request
|
||
|
switch (itemAction) {
|
||
|
|
||
|
case ODA_SELECT:
|
||
|
|
||
|
// Draw/erase selection rect
|
||
|
HandleSelectionState(pDI);
|
||
|
break;
|
||
|
|
||
|
case ODA_FOCUS:
|
||
|
|
||
|
// Draw/erase focus rect
|
||
|
HandleFocusState (pDI);
|
||
|
break;
|
||
|
|
||
|
case ODA_DRAWENTIRE:
|
||
|
|
||
|
// Leave border space for focus rectangle
|
||
|
InflateRect (prect, -OWNER_DRAWN_ITEM, -OWNER_DRAWN_ITEM) ;
|
||
|
|
||
|
// Case on Control ID
|
||
|
switch (CtlID) {
|
||
|
|
||
|
case IDC_LINECOLOR:
|
||
|
|
||
|
// Draw filled rect of item's color
|
||
|
if ( itemID < NumStandardColorIndices() )
|
||
|
Fill(hDC, IndexToStandardColor(itemID), prect);
|
||
|
else
|
||
|
// Custom color item only exists if the currently
|
||
|
// selected item has a custom color defined.
|
||
|
Fill(hDC, m_pInfoSel->Props.rgbColor, prect);
|
||
|
break ;
|
||
|
|
||
|
case IDC_LINEWIDTH:
|
||
|
case IDC_LINESTYLE:
|
||
|
|
||
|
// Clear the item's area
|
||
|
rgbBk = GetSysColor(COLOR_WINDOW);
|
||
|
|
||
|
Fill(hDC, rgbBk, prect);
|
||
|
|
||
|
// Draw centered line showing item's width or style
|
||
|
if (CtlID == IDC_LINEWIDTH)
|
||
|
hPen = CreatePen (PS_SOLID, IndexToWidth(itemID), RGB (0,0,0));
|
||
|
else
|
||
|
hPen = CreatePen (IndexToStyle(itemID), 1, RGB (0,0,0));
|
||
|
|
||
|
if ( NULL != hPen ) {
|
||
|
|
||
|
// Set background to insure dashed lines show properly
|
||
|
rgbOldBk = SetBkColor (hDC, rgbBk) ;
|
||
|
|
||
|
if ( CLR_INVALID != rgbOldBk ) {
|
||
|
|
||
|
Line(hDC, (HPEN)hPen, prect->left + 8,
|
||
|
(prect->top + prect->bottom) / 2,
|
||
|
prect->right - 8,
|
||
|
(prect->top + prect->bottom) / 2);
|
||
|
|
||
|
SetBkColor (hDC, rgbOldBk) ;
|
||
|
}
|
||
|
DeleteObject (hPen) ;
|
||
|
}
|
||
|
break ;
|
||
|
}
|
||
|
|
||
|
// Restore original rect and draw focus/select rects
|
||
|
InflateRect (prect, OWNER_DRAWN_ITEM, OWNER_DRAWN_ITEM) ;
|
||
|
HandleSelectionState (pDI) ;
|
||
|
HandleFocusState (pDI) ;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
INT
|
||
|
CCounterPropPage::ScaleFactorToIndex (
|
||
|
IN INT iScaleFactor
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
ScaleFactorToIndex translates a CounterItem ScaleFactor value to
|
||
|
the appropriate scale factor combo box index.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
iScaleFactor - CounterItem scale factor integer value.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
Scale factor combo box index.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
INT retValue;
|
||
|
|
||
|
if ( INT_MAX == iScaleFactor ) {
|
||
|
retValue = 0;
|
||
|
} else {
|
||
|
retValue = iScaleFactor - PDH_MIN_SCALE + 1;
|
||
|
}
|
||
|
|
||
|
return retValue;
|
||
|
}
|
||
|
|
||
|
INT
|
||
|
CCounterPropPage::IndexToScaleFactor (
|
||
|
IN INT iScaleIndex
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
ScaleFactorToIndex translates a CounterItem ScaleFactor value to
|
||
|
the appropriate scale factor combo box index.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
iScaleIndex - Scale factor combo box index.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
CounterItem scale factor integer value.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
INT retValue;
|
||
|
|
||
|
if ( 0 == iScaleIndex ) {
|
||
|
retValue = INT_MAX;
|
||
|
} else {
|
||
|
retValue = iScaleIndex - 1 + PDH_MIN_SCALE;
|
||
|
}
|
||
|
|
||
|
return retValue;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
CCounterPropPage::SetStyleComboEnable (
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
SetStyleComboEnable enables the style combo box if the width is 1,
|
||
|
disables it otherwise.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
void
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
HWND hWnd;
|
||
|
|
||
|
hWnd = DialogControl(m_hDlg, IDC_CTRLIST);
|
||
|
|
||
|
DialogEnable (m_hDlg, IDC_LABEL_LINESTYLE, (0 == m_props.iWidthIndex) );
|
||
|
if (LBSelectCount(hWnd) == 1) {
|
||
|
DialogEnable (m_hDlg, IDC_LINESTYLE, (0 == m_props.iWidthIndex) );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
HRESULT
|
||
|
CCounterPropPage::EditPropertyImpl( DISPID dispID )
|
||
|
{
|
||
|
HRESULT hr = E_NOTIMPL;
|
||
|
|
||
|
if ( DISPID_VALUE == dispID ) {
|
||
|
m_dwEditControl = IDC_ADDCTR;
|
||
|
hr = S_OK;
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
ULONG
|
||
|
CCounterPropPage::HashCounter(
|
||
|
LPWSTR szCounterName
|
||
|
)
|
||
|
{
|
||
|
ULONG h = 0;
|
||
|
ULONG a = 31415; //a, b, k are primes
|
||
|
const ULONG k = 16381;
|
||
|
const ULONG b = 27183;
|
||
|
LPWSTR szThisChar;
|
||
|
WCHAR Char;
|
||
|
|
||
|
if (szCounterName) {
|
||
|
for (szThisChar = szCounterName; * szThisChar; szThisChar ++) {
|
||
|
Char = * szThisChar;
|
||
|
if (_istupper(Char) ) {
|
||
|
Char = _tolower(Char);
|
||
|
}
|
||
|
h = (a * h + ((ULONG) Char)) % k;
|
||
|
a = a * b % (k - 1);
|
||
|
}
|
||
|
}
|
||
|
return (h % eHashTableSize);
|
||
|
}
|
||
|
|
||
|
|
||
|
//++
|
||
|
// Description:
|
||
|
// Remove a counter path from hash table. One counter
|
||
|
// path must exactly match the given one in order to be
|
||
|
// removed, even it is one with wildcard
|
||
|
//
|
||
|
// Parameters:
|
||
|
// pItemInfo - Pointer to item info to be removed
|
||
|
//
|
||
|
// Return:
|
||
|
// Return TRUE if the counter path is removed, otherwis return FALSE
|
||
|
//--
|
||
|
BOOL
|
||
|
CCounterPropPage::RemoveCounterFromHashTable(
|
||
|
LPWSTR pszPath,
|
||
|
PPDH_COUNTER_PATH_ELEMENTS pCounter
|
||
|
)
|
||
|
{
|
||
|
ULONG lHashValue;
|
||
|
PHASH_ENTRY pEntry = NULL;
|
||
|
PHASH_ENTRY pPrev = NULL;
|
||
|
BOOL bReturn = FALSE;
|
||
|
LPWSTR pszFullPath = NULL;
|
||
|
|
||
|
SetLastError(ERROR_SUCCESS);
|
||
|
|
||
|
if (pszPath == NULL || pCounter == NULL) {
|
||
|
SetLastError(ERROR_INVALID_PARAMETER);
|
||
|
goto ErrorOut;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// If the counter path does not have machine name,
|
||
|
// add the machine name to compose a full path
|
||
|
//
|
||
|
if (*pszPath == L'\\' && (*(pszPath+1) == L'\\') ) {
|
||
|
lHashValue = HashCounter(pszPath);
|
||
|
}
|
||
|
else {
|
||
|
ULONG ulBufLen;
|
||
|
|
||
|
if (pCounter->szMachineName == NULL) {
|
||
|
SetLastError(ERROR_INVALID_PARAMETER);
|
||
|
goto ErrorOut;
|
||
|
}
|
||
|
|
||
|
ulBufLen = lstrlen(pCounter->szMachineName) + lstrlen(pszPath) + 1;
|
||
|
|
||
|
pszFullPath = new WCHAR [ ulBufLen];
|
||
|
|
||
|
if (pszFullPath == NULL) {
|
||
|
SetLastError(ERROR_OUTOFMEMORY);
|
||
|
goto ErrorOut;
|
||
|
}
|
||
|
StringCchCopy(pszFullPath, ulBufLen, pCounter->szMachineName);
|
||
|
StringCchCat(pszFullPath, ulBufLen, pszPath);
|
||
|
|
||
|
lHashValue = HashCounter(pszFullPath);
|
||
|
}
|
||
|
|
||
|
pEntry = m_HashTable[lHashValue];
|
||
|
|
||
|
//
|
||
|
// Check if there is a counter path which is exactly the same
|
||
|
// as the given one
|
||
|
//
|
||
|
while (pEntry) {
|
||
|
if (pEntry->pCounter == pCounter)
|
||
|
break;
|
||
|
pPrev = pEntry;
|
||
|
pEntry = pEntry->pNext;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// If we found it, remove it
|
||
|
//
|
||
|
if (pEntry) {
|
||
|
if (pPrev == NULL) {
|
||
|
m_HashTable[lHashValue] = pEntry->pNext;
|
||
|
}
|
||
|
else {
|
||
|
pPrev->pNext = pEntry->pNext;
|
||
|
}
|
||
|
assert (pEntry->pCounter);
|
||
|
delete(pEntry->pCounter);
|
||
|
delete(pEntry);
|
||
|
|
||
|
bReturn = TRUE;
|
||
|
}
|
||
|
|
||
|
ErrorOut:
|
||
|
if (pszFullPath != NULL) {
|
||
|
delete [] pszFullPath;
|
||
|
}
|
||
|
|
||
|
return bReturn;
|
||
|
}
|
||
|
|
||
|
|
||
|
//++
|
||
|
// Description:
|
||
|
// Insert a counter path into hash table.
|
||
|
//
|
||
|
// Parameters:
|
||
|
// PItemInfo - Pointer to the counter item info
|
||
|
//
|
||
|
// Return:
|
||
|
// Return the pointer to new inserted PDH_COUNTER_PATH_ELEMENTS structure
|
||
|
//--
|
||
|
|
||
|
DWORD
|
||
|
CCounterPropPage::InsertCounterToHashTable(
|
||
|
LPWSTR pszPath,
|
||
|
PPDH_COUNTER_PATH_ELEMENTS* ppCounter
|
||
|
)
|
||
|
{
|
||
|
ULONG lHashValue;
|
||
|
PHASH_ENTRY pEntry = NULL;
|
||
|
PHASH_ENTRY pPrev = NULL;
|
||
|
PDH_STATUS pdhStatus;
|
||
|
ULONG ulBufSize = 0;
|
||
|
PPDH_COUNTER_PATH_ELEMENTS pCounter = NULL;
|
||
|
LPWSTR pszFullPath = NULL;
|
||
|
BOOL bExisted = FALSE;
|
||
|
DWORD dwResult;
|
||
|
|
||
|
dwResult = ERROR_SUCCESS;
|
||
|
|
||
|
if (pszPath == NULL || ppCounter == NULL) {
|
||
|
dwResult = ERROR_INVALID_PARAMETER;
|
||
|
goto ErrorOut;
|
||
|
}
|
||
|
|
||
|
*ppCounter = NULL;
|
||
|
|
||
|
//
|
||
|
// Parse the counter path
|
||
|
//
|
||
|
pCounter = NULL;
|
||
|
do {
|
||
|
if (pCounter) {
|
||
|
delete [] (char*)pCounter;
|
||
|
pCounter = NULL;
|
||
|
}
|
||
|
else {
|
||
|
ulBufSize = sizeof(PDH_COUNTER_PATH_ELEMENTS) + sizeof(WCHAR) * PDH_MAX_COUNTER_PATH;
|
||
|
}
|
||
|
|
||
|
pCounter = (PPDH_COUNTER_PATH_ELEMENTS) new char [ ulBufSize ];
|
||
|
|
||
|
if (pCounter == NULL) {
|
||
|
dwResult = ERROR_OUTOFMEMORY;
|
||
|
goto ErrorOut;
|
||
|
}
|
||
|
|
||
|
pdhStatus = PdhParseCounterPath( pszPath, pCounter, & ulBufSize, 0);
|
||
|
|
||
|
} while (pdhStatus == PDH_INSUFFICIENT_BUFFER || pdhStatus == PDH_MORE_DATA);
|
||
|
|
||
|
if (pdhStatus != ERROR_SUCCESS) {
|
||
|
dwResult = pdhStatus;
|
||
|
goto ErrorOut;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// If the counter path does not have machine name,
|
||
|
// add the machine name to compose a full path
|
||
|
//
|
||
|
if (*pszPath == L'\\' && (*(pszPath+1) == L'\\') ) {
|
||
|
lHashValue = HashCounter(pszPath);
|
||
|
}
|
||
|
else {
|
||
|
ULONG ulBufLen;
|
||
|
|
||
|
ulBufLen = lstrlen(pCounter->szMachineName) + lstrlen(pszPath) + 1;
|
||
|
pszFullPath = new WCHAR [ulBufLen];
|
||
|
|
||
|
if (pszFullPath == NULL) {
|
||
|
dwResult = ERROR_OUTOFMEMORY;
|
||
|
goto ErrorOut;
|
||
|
}
|
||
|
StringCchCopy(pszFullPath, ulBufLen, pCounter->szMachineName);
|
||
|
StringCchCat(pszFullPath, ulBufLen, pszPath);
|
||
|
|
||
|
lHashValue = HashCounter(pszFullPath);
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Check if there is a counter path which is exactly the same
|
||
|
// as the given one
|
||
|
//
|
||
|
pEntry = m_HashTable[lHashValue];
|
||
|
|
||
|
while (pEntry) {
|
||
|
if ( AreSameCounterPath ( pEntry->pCounter, pCounter ) ) {
|
||
|
dwResult = SMON_STATUS_DUPL_COUNTER_PATH;
|
||
|
bExisted = TRUE;
|
||
|
*ppCounter = pEntry->pCounter;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
pPrev = pEntry;
|
||
|
pEntry = pEntry->pNext;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// Add the new counter path
|
||
|
//
|
||
|
if (bExisted == FALSE) {
|
||
|
pEntry = (PHASH_ENTRY) new HASH_ENTRY;
|
||
|
if (pEntry == NULL) {
|
||
|
dwResult = ERROR_OUTOFMEMORY;
|
||
|
goto ErrorOut;
|
||
|
}
|
||
|
|
||
|
pEntry->pCounter = pCounter;
|
||
|
pEntry->pNext = m_HashTable[lHashValue];
|
||
|
m_HashTable[lHashValue] = pEntry;
|
||
|
*ppCounter = pCounter;
|
||
|
}
|
||
|
|
||
|
if (pszFullPath != NULL) {
|
||
|
delete [] pszFullPath;
|
||
|
}
|
||
|
return dwResult;
|
||
|
|
||
|
ErrorOut:
|
||
|
if (pszFullPath != NULL) {
|
||
|
delete [] pszFullPath;
|
||
|
}
|
||
|
|
||
|
if (pCounter != NULL) {
|
||
|
delete [] ((char*) pCounter);
|
||
|
}
|
||
|
|
||
|
return dwResult;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
//++
|
||
|
// Description:
|
||
|
// The function clears all the entries in hash table
|
||
|
// and set hash-table-not-set-up flag
|
||
|
//
|
||
|
// Parameters:
|
||
|
// None
|
||
|
//
|
||
|
// Return:
|
||
|
// None
|
||
|
//--
|
||
|
void
|
||
|
CCounterPropPage::ClearCountersHashTable( void )
|
||
|
{
|
||
|
ULONG i;
|
||
|
PHASH_ENTRY pEntry;
|
||
|
PHASH_ENTRY pNext;
|
||
|
|
||
|
if (m_fHashTableSetup) {
|
||
|
for (i = 0; i < eHashTableSize; i ++) {
|
||
|
pNext = m_HashTable[i];
|
||
|
while (pNext != NULL) {
|
||
|
pEntry = pNext;
|
||
|
pNext = pEntry->pNext;
|
||
|
|
||
|
assert( pEntry->pCounter);
|
||
|
|
||
|
delete pEntry->pCounter;
|
||
|
delete (pEntry);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
memset(&m_HashTable, 0, sizeof(m_HashTable));
|
||
|
}
|
||
|
m_fHashTableSetup = FALSE;
|
||
|
}
|
||
|
|
||
|
void CCounterPropPage::DeleteInfo(PItemInfo pInfo)
|
||
|
{
|
||
|
if (pInfo == NULL) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (pInfo->pszPath != NULL) {
|
||
|
delete (pInfo->pszPath);
|
||
|
}
|
||
|
if ( pInfo->pItem != NULL ) {
|
||
|
pInfo->pItem->Release();
|
||
|
}
|
||
|
|
||
|
delete pInfo;
|
||
|
}
|