437 lines
12 KiB
C++
437 lines
12 KiB
C++
/*++
|
|
|
|
Copyright (C) 1996-1999 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
statbar.cpp
|
|
|
|
Abstract:
|
|
|
|
Implementation of the value bar class.
|
|
|
|
--*/
|
|
|
|
#include "polyline.h"
|
|
#include <strsafe.h>
|
|
#include "winhelpr.h"
|
|
#include "grphitem.h"
|
|
#include "statbar.h"
|
|
|
|
#define MAX_STAT_LABEL_LEN 32
|
|
#define INVALID_VALUE (-1.0)
|
|
#define szDashLine L"--------- "
|
|
|
|
//
|
|
// ??? Why do we put these outside of the class ???
|
|
//
|
|
static WCHAR aszItemLabel[STAT_ITEM_CNT][MAX_STAT_LABEL_LEN];
|
|
|
|
static BOOLEAN fInitDone = FALSE;
|
|
|
|
CStatsBar::CStatsBar(void)
|
|
: m_pCtrl ( NULL ),
|
|
m_iFontHeight ( 0 ),
|
|
m_iValueWidth ( 0 ),
|
|
m_pGraphItemToInit ( NULL )
|
|
{
|
|
memset (&m_Rect, 0, sizeof(m_Rect));
|
|
}
|
|
|
|
CStatsBar::~CStatsBar(void)
|
|
{
|
|
}
|
|
|
|
BOOL CStatsBar::Init (
|
|
PSYSMONCTRL pCtrl,
|
|
HWND /* hWnd */
|
|
)
|
|
{
|
|
INT i;
|
|
|
|
// save pointer to primary object
|
|
m_pCtrl = pCtrl;
|
|
|
|
// First time through, load the item labels
|
|
if (!fInitDone) {
|
|
fInitDone = TRUE;
|
|
|
|
for (i = 0; i < STAT_ITEM_CNT; i++) {
|
|
LoadString(g_hInstance, (IDS_STAT_BASE + i), aszItemLabel[i], MAX_STAT_LABEL_LEN);
|
|
}
|
|
}
|
|
|
|
// Initialze the stat values
|
|
Clear();
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
void CStatsBar::SizeComponents(LPRECT pRect)
|
|
{
|
|
// Just save the rectangle
|
|
m_Rect = *pRect;
|
|
}
|
|
|
|
void CStatsBar::SetTimeSpan(double dSeconds)
|
|
{
|
|
m_StatItem[STAT_TIME].dNewValue = dSeconds;
|
|
}
|
|
|
|
|
|
void CStatsBar::Update(HDC hDC, PCGraphItem pGraphItem)
|
|
{
|
|
double dMin, dMax, dAvg, dVal;
|
|
PSTAT_ITEM pItem;
|
|
HRESULT hr;
|
|
PDH_STATUS stat;
|
|
LONG lCtrStat;
|
|
INT i;
|
|
|
|
// if no space assigned, return
|
|
if (m_Rect.top == m_Rect.bottom) {
|
|
m_pGraphItemToInit = pGraphItem;
|
|
m_StatItem[0].iInitialized = 0;
|
|
return;
|
|
}
|
|
|
|
if (pGraphItem == NULL) {
|
|
m_pGraphItemToInit = NULL;
|
|
pItem = &m_StatItem[0];
|
|
for (i=0; i<STAT_ITEM_CNT-1; i++, pItem++) {
|
|
pItem->dNewValue = INVALID_VALUE;
|
|
pItem->iInitialized = 0;
|
|
}
|
|
} else {
|
|
|
|
stat = pGraphItem->GetValue(&dVal, &lCtrStat);
|
|
if (stat == 0 && IsSuccessSeverity(lCtrStat))
|
|
m_StatItem[STAT_LAST].dNewValue = dVal;
|
|
else
|
|
m_StatItem[STAT_LAST].dNewValue = INVALID_VALUE;
|
|
|
|
hr = pGraphItem->GetStatistics(&dMax, &dMin, &dAvg, &lCtrStat);
|
|
if (SUCCEEDED(hr) && IsSuccessSeverity(lCtrStat)) {
|
|
m_StatItem[STAT_MIN].dNewValue = dMin;
|
|
m_StatItem[STAT_MAX].dNewValue = dMax;
|
|
m_StatItem[STAT_AVG].dNewValue = dAvg;
|
|
} else {
|
|
m_StatItem[STAT_MIN].dNewValue = INVALID_VALUE;
|
|
m_StatItem[STAT_MAX].dNewValue = INVALID_VALUE;
|
|
m_StatItem[STAT_AVG].dNewValue = INVALID_VALUE;
|
|
}
|
|
m_StatItem[0].dwCounterType = pGraphItem->m_CounterInfo.dwType;
|
|
m_StatItem[0].iInitialized = 1;
|
|
}
|
|
|
|
// hDC is null if updating only values.
|
|
if (hDC != NULL) {
|
|
SetBkColor(hDC, m_pCtrl->clrBackCtl());
|
|
SetTextColor(hDC, m_pCtrl->clrFgnd());
|
|
DrawValues(hDC,FALSE);
|
|
}
|
|
}
|
|
|
|
void CStatsBar::Clear( void )
|
|
{
|
|
INT i;
|
|
|
|
for (i = 0; i < STAT_ITEM_CNT-1; i++) {
|
|
m_StatItem[i].dValue = INVALID_VALUE;
|
|
m_StatItem[i].dNewValue = INVALID_VALUE;
|
|
m_StatItem[i].iInitialized = 0;
|
|
}
|
|
}
|
|
|
|
void CStatsBar::Draw (HDC hDC, HDC /* hAttribDC */, PRECT prcUpdate)
|
|
{
|
|
RECT rectFrame;
|
|
PSTAT_ITEM pItem;
|
|
HFONT hFontOld;
|
|
INT i;
|
|
RECT rectPaint;
|
|
RECT rectClip;
|
|
|
|
// if no space assigned, return
|
|
if (m_Rect.top == m_Rect.bottom)
|
|
return;
|
|
|
|
// if no painting needed, return
|
|
if (!IntersectRect(&rectPaint, &m_Rect, prcUpdate))
|
|
return;
|
|
|
|
SetBkMode(hDC, TRANSPARENT);
|
|
SetTextColor(hDC, m_pCtrl->clrFgnd());
|
|
SetTextAlign(hDC, TA_LEFT|TA_TOP);
|
|
|
|
hFontOld = SelectFont(hDC, m_pCtrl->Font());
|
|
|
|
pItem = &m_StatItem[0];
|
|
|
|
// If the stat bar was hidden on Update, for example if
|
|
// the control was loaded from a property bag, or if a
|
|
// counter was selected while the stat bar was hidden,
|
|
// initialize it here.
|
|
if ( 0 == pItem->iInitialized ) {
|
|
Update ( NULL, m_pGraphItemToInit );
|
|
}
|
|
|
|
// Draw Label and 3D box for each item
|
|
for (i=0; i<STAT_ITEM_CNT; i++, pItem++) {
|
|
|
|
rectClip.top = m_Rect.top + pItem->yPos + RECT_BORDER;
|
|
rectClip.bottom = rectClip.top + m_iFontHeight;
|
|
rectClip.left = m_Rect.left + pItem->xPos;
|
|
rectClip.right = rectClip.left + pItem->xLabelWidth;
|
|
|
|
ExtTextOut(
|
|
hDC,
|
|
m_Rect.left + pItem->xPos,
|
|
m_Rect.top + pItem->yPos + RECT_BORDER,
|
|
0,
|
|
&rectClip,
|
|
aszItemLabel[i],
|
|
lstrlen(aszItemLabel[i]),
|
|
NULL );
|
|
|
|
if ( eAppear3D == m_pCtrl->Appearance() ) {
|
|
rectFrame.left = m_Rect.left + pItem->xPos + pItem->xLabelWidth + VALUE_MARGIN;
|
|
rectFrame.right = rectFrame.left + m_iValueWidth + 2 * RECT_BORDER;
|
|
rectFrame.top = m_Rect.top + pItem->yPos;
|
|
rectFrame.bottom = rectFrame.top + m_iFontHeight + 2 * RECT_BORDER;
|
|
DrawEdge(hDC, &rectFrame, BDR_SUNKENOUTER, BF_RECT);
|
|
}
|
|
}
|
|
|
|
SelectFont(hDC, hFontOld);
|
|
|
|
SetBkMode(hDC, OPAQUE);
|
|
SetBkColor(hDC, m_pCtrl->clrBackCtl());
|
|
DrawValues(hDC, TRUE);
|
|
}
|
|
|
|
|
|
void CStatsBar::DrawValues(HDC hDC, BOOL bForce)
|
|
{
|
|
RECT rectValue ;
|
|
WCHAR szValue [MAX_VALUE_LEN] ;
|
|
HFONT hFontOld;
|
|
PSTAT_ITEM pItem = NULL;
|
|
INT i;
|
|
INT nSecs, nMins, nHours, nDays;
|
|
|
|
SetTextAlign(hDC, TA_RIGHT | TA_TOP);
|
|
hFontOld = SelectFont(hDC, m_pCtrl->Font());
|
|
|
|
pItem = &m_StatItem[0];
|
|
|
|
for (i=0; i<STAT_ITEM_CNT; i++,pItem++) {
|
|
|
|
if ( NULL == pItem )
|
|
continue;
|
|
|
|
if ((pItem->dValue == pItem->dNewValue) && !bForce)
|
|
continue;
|
|
|
|
pItem->dValue = pItem->dNewValue;
|
|
|
|
rectValue.top = m_Rect.top + pItem->yPos + RECT_BORDER;
|
|
rectValue.bottom = rectValue.top + m_iFontHeight;
|
|
rectValue.left = m_Rect.left + pItem->xPos + pItem->xLabelWidth + VALUE_MARGIN + RECT_BORDER;
|
|
rectValue.right = rectValue.left + m_iValueWidth - 1;
|
|
|
|
if (i == STAT_TIME) {
|
|
LPWSTR pszTimeSep = NULL;
|
|
|
|
pszTimeSep = GetTimeSeparator ( );
|
|
|
|
nSecs = (INT)pItem->dValue;
|
|
|
|
nMins = nSecs / 60;
|
|
nSecs -= nMins * 60;
|
|
|
|
nHours = nMins / 60;
|
|
nMins -= nHours * 60;
|
|
|
|
nDays = nHours / 24;
|
|
nHours -= nDays * 24;
|
|
|
|
if (nDays != 0) {
|
|
StringCchPrintf(szValue, MAX_VALUE_LEN, SZ_DAYTIME_FORMAT, nDays, nHours, pszTimeSep, nMins);
|
|
} else {
|
|
if (nHours != 0)
|
|
StringCchPrintf(szValue, MAX_VALUE_LEN, SZ_HRTIME_FORMAT, nHours, pszTimeSep, nMins, pszTimeSep, nSecs);
|
|
else
|
|
StringCchPrintf(szValue, MAX_VALUE_LEN, SZ_MINTIME_FORMAT, nMins, pszTimeSep, nSecs);
|
|
}
|
|
} else {
|
|
|
|
if ( INVALID_VALUE == pItem->dValue ) {
|
|
StringCchCopy ( szValue, MAX_VALUE_LEN, szDashLine );
|
|
} else if (pItem->dValue > E_MEDIUM_VALUE) {
|
|
if (pItem->dValue > E_TOO_LARGE_VALUE) {
|
|
StringCchCopy(szValue, MAX_VALUE_LEN, SZ_VALUE_TOO_HIGH) ;
|
|
} else {
|
|
|
|
if ( pItem->dValue <= E_LARGE_VALUE ) {
|
|
|
|
FormatNumber (
|
|
pItem->dValue,
|
|
szValue,
|
|
MAX_VALUE_LEN,
|
|
eMinimumWidth,
|
|
eMediumPrecision );
|
|
|
|
} else {
|
|
|
|
FormatScientific (
|
|
pItem->dValue,
|
|
szValue,
|
|
MAX_VALUE_LEN,
|
|
eMinimumWidth,
|
|
eLargePrecision );
|
|
}
|
|
}
|
|
|
|
} else if (pItem->dValue < -E_MEDIUM_VALUE) {
|
|
if (pItem->dValue < -E_TOO_LARGE_VALUE) {
|
|
StringCchCopy(szValue, MAX_VALUE_LEN, SZ_VALUE_TOO_LOW) ;
|
|
} else {
|
|
if ( pItem->dValue >= -E_LARGE_VALUE ) {
|
|
FormatNumber (
|
|
pItem->dValue,
|
|
szValue,
|
|
MAX_VALUE_LEN,
|
|
eMinimumWidth,
|
|
eMediumPrecision );
|
|
} else {
|
|
FormatScientific (
|
|
pItem->dValue,
|
|
szValue,
|
|
MAX_VALUE_LEN,
|
|
eMinimumWidth,
|
|
eLargePrecision );
|
|
}
|
|
}
|
|
} else {
|
|
if ( ( m_StatItem[0].dwCounterType &
|
|
( PERF_TYPE_COUNTER | PERF_TYPE_TEXT ) ) ) {
|
|
FormatNumber (
|
|
pItem->dValue,
|
|
szValue,
|
|
MAX_VALUE_LEN,
|
|
eMinimumWidth,
|
|
eSmallPrecision );
|
|
} else {
|
|
FormatNumber (
|
|
pItem->dValue,
|
|
szValue,
|
|
MAX_VALUE_LEN,
|
|
eMinimumWidth,
|
|
eIntegerPrecision );
|
|
}
|
|
}
|
|
}
|
|
|
|
// TextOut (hDC, rectValue.right, rectValue.top, szValue, lstrlen (szValue)) ;
|
|
|
|
ExtTextOut (hDC, rectValue.right, rectValue.top, ETO_OPAQUE, &rectValue,
|
|
szValue, lstrlen (szValue), NULL) ;
|
|
}
|
|
|
|
SelectFont(hDC, hFontOld);
|
|
}
|
|
|
|
|
|
|
|
INT CStatsBar::Height (INT iMaxHeight, INT iMaxWidth)
|
|
{
|
|
INT iHeight;
|
|
INT xPos,yPos;
|
|
PSTAT_ITEM pItem;
|
|
INT i,j;
|
|
INT iItemWidth;
|
|
INT iFirst;
|
|
INT iRemainder;
|
|
|
|
iMaxWidth -= 2 * RECT_BORDER;
|
|
xPos = 0;
|
|
yPos = 0;
|
|
iFirst = 0;
|
|
pItem = &m_StatItem[0];
|
|
|
|
for (i=0; i<STAT_ITEM_CNT; i++,pItem++) {
|
|
|
|
iItemWidth = pItem->xLabelWidth + VALUE_MARGIN + m_iValueWidth;
|
|
if (iItemWidth > iMaxWidth)
|
|
return 0;
|
|
|
|
if (xPos + iItemWidth > iMaxWidth) {
|
|
iRemainder = iMaxWidth - xPos + LABEL_MARGIN;
|
|
xPos = 0;
|
|
yPos += m_iFontHeight + LINE_SPACING;
|
|
|
|
for (j=iFirst; j<i; j++) {
|
|
m_StatItem[j].xPos += iRemainder;
|
|
}
|
|
iFirst = i;
|
|
}
|
|
|
|
pItem->xPos = xPos;
|
|
pItem->yPos = yPos;
|
|
xPos += (iItemWidth + LABEL_MARGIN);
|
|
}
|
|
|
|
iRemainder = (iMaxWidth - xPos) + LABEL_MARGIN;
|
|
for (j=iFirst; j<STAT_ITEM_CNT; j++) {
|
|
m_StatItem[j].xPos += iRemainder;
|
|
}
|
|
|
|
// if allowed height is not enough, return zero
|
|
iHeight = yPos + m_iFontHeight + 2 * RECT_BORDER;
|
|
|
|
return (iHeight <= iMaxHeight) ? iHeight : 0;
|
|
}
|
|
|
|
|
|
|
|
void CStatsBar::ChangeFont(
|
|
HDC hDC
|
|
)
|
|
{
|
|
INT xPos,yPos;
|
|
WCHAR szValue[MAX_VALUE_LEN];
|
|
HFONT hFontOld;
|
|
PSTAT_ITEM pItem;
|
|
INT i;
|
|
SIZE size;
|
|
|
|
hFontOld = (HFONT)SelectFont(hDC, m_pCtrl->Font());
|
|
|
|
// Get width/height of longest value string
|
|
FormatNumber (
|
|
E_LARGE_VALUE,
|
|
szValue,
|
|
MAX_VALUE_LEN,
|
|
eMinimumWidth,
|
|
eLargePrecision );
|
|
|
|
GetTextExtentPoint32(hDC, szValue, lstrlen(szValue), &size);
|
|
m_iValueWidth = size.cx;
|
|
m_iFontHeight = size.cy;
|
|
|
|
// Do for all stat items
|
|
xPos = 0;
|
|
yPos = 0;
|
|
pItem = &m_StatItem[0];
|
|
|
|
for (i=0; i<STAT_ITEM_CNT; i++,pItem++) {
|
|
if ( NULL != pItem ) {
|
|
pItem->xLabelWidth = TextWidth(hDC, aszItemLabel[i]);
|
|
}
|
|
}
|
|
|
|
SelectFont(hDC, hFontOld);
|
|
}
|