mirror of
https://github.com/nillerusr/source-engine.git
synced 2025-01-10 09:26:43 +00:00
247 lines
5.7 KiB
C++
247 lines
5.7 KiB
C++
|
//========= Copyright Valve Corporation, All rights reserved. ============//
|
||
|
//
|
||
|
// Purpose:
|
||
|
//
|
||
|
// $NoKeywords: $
|
||
|
//
|
||
|
//=============================================================================//
|
||
|
// GraphControl.cpp : implementation file
|
||
|
//
|
||
|
|
||
|
#include "stdafx.h"
|
||
|
#include "vmpi_browser_job_watch.h"
|
||
|
#include "GraphControl.h"
|
||
|
#include "mathlib/mathlib.h"
|
||
|
|
||
|
#ifdef _DEBUG
|
||
|
#define new DEBUG_NEW
|
||
|
#undef THIS_FILE
|
||
|
static char THIS_FILE[] = __FILE__;
|
||
|
#endif
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// CGraphControl
|
||
|
|
||
|
CGraphControl::CGraphControl()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
CGraphControl::~CGraphControl()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
|
||
|
BEGIN_MESSAGE_MAP(CGraphControl, CWnd)
|
||
|
//{{AFX_MSG_MAP(CGraphControl)
|
||
|
ON_WM_PAINT()
|
||
|
//}}AFX_MSG_MAP
|
||
|
END_MESSAGE_MAP()
|
||
|
|
||
|
|
||
|
void CGraphControl::Clear()
|
||
|
{
|
||
|
CRect rcClient;
|
||
|
GetClientRect( rcClient );
|
||
|
|
||
|
CDC *pDC = GetDC();
|
||
|
|
||
|
CBrush brush( RGB( 0, 0, 0 ) );
|
||
|
CBrush *pOldBrush = pDC->SelectObject( &brush );
|
||
|
pDC->Rectangle( 0, 0, rcClient.Width(), rcClient.Height() );
|
||
|
pDC->SelectObject( pOldBrush );
|
||
|
|
||
|
ReleaseDC( pDC );
|
||
|
}
|
||
|
|
||
|
void CGraphControl::Render( CDC *pDC )
|
||
|
{
|
||
|
// Clear the background.
|
||
|
CRect rcClient;
|
||
|
GetClientRect( rcClient );
|
||
|
|
||
|
|
||
|
CBrush brush( RGB( 0, 0, 0 ) );
|
||
|
CBrush *pOldBrush = pDC->SelectObject( &brush );
|
||
|
pDC->Rectangle( 0, 0, rcClient.Width(), rcClient.Height() );
|
||
|
pDC->SelectObject( pOldBrush );
|
||
|
|
||
|
|
||
|
|
||
|
// Work backwards from the right side to the left.
|
||
|
int nIntervals = rcClient.Width();
|
||
|
DWORD intervalMS = 500; // one interval per pixel
|
||
|
DWORD startTime = 0xFFFFFFFF, endTime = 0;
|
||
|
|
||
|
// First, find which order of magnitude to use on the vertical scale by finding the maximum value.
|
||
|
for ( int iEntry=0; iEntry < m_Entries.Count(); iEntry++ )
|
||
|
{
|
||
|
DWORD msTime = m_Entries[iEntry].m_msTime;
|
||
|
startTime = min( startTime, msTime );
|
||
|
endTime = max( endTime, msTime );
|
||
|
}
|
||
|
|
||
|
int curTime = (int)endTime - nIntervals*intervalMS;
|
||
|
|
||
|
|
||
|
|
||
|
CGraphEntry prevEntry, curEntry;
|
||
|
prevEntry.m_msTime = curEntry.m_msTime = -1;
|
||
|
|
||
|
CUtlVector<POINT> sentPoints;
|
||
|
CUtlVector<POINT> receivedPoints;
|
||
|
|
||
|
int iCurEntry = -1;
|
||
|
int nMaxBytesSent = -1, nMaxBytesReceived = -1;
|
||
|
|
||
|
for ( int x=0; x < nIntervals; x++ )
|
||
|
{
|
||
|
if ( curTime >= 0 )
|
||
|
{
|
||
|
// Now find the graph_entry for the time we're at.
|
||
|
while ( prevEntry.m_msTime == -1 || curTime > curEntry.m_msTime )
|
||
|
{
|
||
|
++iCurEntry;
|
||
|
if ( iCurEntry >= m_Entries.Count() )
|
||
|
goto ENDLOOP;
|
||
|
|
||
|
prevEntry = curEntry;
|
||
|
curEntry = m_Entries[iCurEntry];
|
||
|
}
|
||
|
|
||
|
if ( curTime >= prevEntry.m_msTime && curTime <= curEntry.m_msTime )
|
||
|
{
|
||
|
// Interpolate the bytes sent.
|
||
|
int nBytesSent = (int)RemapVal(
|
||
|
curTime,
|
||
|
prevEntry.m_msTime, curEntry.m_msTime,
|
||
|
prevEntry.m_nBytesSent, curEntry.m_nBytesSent );
|
||
|
|
||
|
POINT sentPoint = { x, nBytesSent };
|
||
|
sentPoints.AddToTail( sentPoint );
|
||
|
nMaxBytesSent = max( nMaxBytesSent, nBytesSent );
|
||
|
|
||
|
|
||
|
int nBytesReceived = (int)RemapVal(
|
||
|
curTime,
|
||
|
prevEntry.m_msTime, curEntry.m_msTime,
|
||
|
prevEntry.m_nBytesReceived, curEntry.m_nBytesReceived );
|
||
|
|
||
|
POINT receivedPoint = { x, nBytesReceived };
|
||
|
receivedPoints.AddToTail( receivedPoint );
|
||
|
nMaxBytesReceived = max( nMaxBytesReceived, nBytesReceived );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
curTime += intervalMS;
|
||
|
}
|
||
|
|
||
|
ENDLOOP:;
|
||
|
|
||
|
|
||
|
// Now normalize all the values.
|
||
|
int largest = max( nMaxBytesSent, nMaxBytesReceived );
|
||
|
int topValue = (largest*11) / 10;
|
||
|
/*
|
||
|
DWORD nZeros;
|
||
|
for( nZeros = 1; nZeros < 20; nZeros++ )
|
||
|
{
|
||
|
if ( largest < pow( 10, nZeros ) )
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// Now find the value at the top of the graph. We choose the smallest enclosing tenth of the
|
||
|
// order of magnitude we're at (so if we were at 1,000,000, and our max value was 350,000, we'd choose 400,000).
|
||
|
int iTenth;
|
||
|
int topValue;
|
||
|
for ( iTenth=1; iTenth <= 10; iTenth++ )
|
||
|
{
|
||
|
topValue = (DWORD)( pow( 10, nZeros-1 ) * iTenth );
|
||
|
if ( topValue >= largest )
|
||
|
break;
|
||
|
}
|
||
|
*/
|
||
|
|
||
|
for ( int iSample=0; iSample < sentPoints.Count(); iSample++ )
|
||
|
{
|
||
|
double flHeight;
|
||
|
|
||
|
flHeight = ((double)sentPoints[iSample].y / topValue) * (rcClient.Height() - 1);
|
||
|
sentPoints[iSample].y = (int)( rcClient.Height() - flHeight );
|
||
|
|
||
|
flHeight = ((double)receivedPoints[iSample].y / topValue) * (rcClient.Height() - 1);
|
||
|
receivedPoints[iSample].y = (int)( rcClient.Height() - flHeight );
|
||
|
}
|
||
|
|
||
|
|
||
|
// Draw some horizontal lines dividing the space.
|
||
|
int nLines = 10;
|
||
|
for ( int iLine=0; iLine <= nLines; iLine++ )
|
||
|
{
|
||
|
CPen penLine;
|
||
|
COLORREF color;
|
||
|
if ( iLine == 0 || iLine == nLines/2 )
|
||
|
color = RGB( 0, 220, 0 );
|
||
|
else
|
||
|
color = RGB( 0, 100, 0 );
|
||
|
|
||
|
penLine.CreatePen( PS_SOLID, 1, color );
|
||
|
CPen *pOldPen = pDC->SelectObject( &penLine );
|
||
|
|
||
|
int y = (iLine * rcClient.Height()) / nLines;
|
||
|
pDC->MoveTo( 0, y );
|
||
|
pDC->LineTo( rcClient.Width(), y );
|
||
|
|
||
|
pDC->SelectObject( pOldPen );
|
||
|
}
|
||
|
|
||
|
|
||
|
// Now draw the lines for the data.
|
||
|
CPen penSent( PS_SOLID, 1, RGB( 0, 255, 0 ) );
|
||
|
CPen *pOldPen = pDC->SelectObject( &penSent );
|
||
|
pDC->Polyline( sentPoints.Base(), sentPoints.Count() );
|
||
|
pDC->SelectObject( pOldPen );
|
||
|
|
||
|
CPen penReceived( PS_SOLID, 1, RGB( 255, 255, 0 ) );
|
||
|
pOldPen = pDC->SelectObject( &penReceived );
|
||
|
pDC->Polyline( receivedPoints.Base(), receivedPoints.Count() );
|
||
|
pDC->SelectObject( pOldPen );
|
||
|
|
||
|
|
||
|
|
||
|
// Draw text labels.
|
||
|
pDC->SetTextColor( RGB( 200, 200, 200 ) );
|
||
|
pDC->SetBkColor( 0 );
|
||
|
|
||
|
CString str;
|
||
|
str.Format( "%dk", (topValue+511) / 1024 );
|
||
|
pDC->ExtTextOut( 0, 1, 0, NULL, str, NULL );
|
||
|
|
||
|
str.Format( "%dk", (topValue+511) / 1024 / 2 );
|
||
|
pDC->ExtTextOut( 0, rcClient.Height()/2 + 1, 0, NULL, str, NULL );
|
||
|
}
|
||
|
|
||
|
|
||
|
void CGraphControl::Fill( CUtlVector<CGraphEntry> &entries )
|
||
|
{
|
||
|
CDC *pDC = GetDC();
|
||
|
if ( !pDC )
|
||
|
return;
|
||
|
|
||
|
m_Entries = entries;
|
||
|
|
||
|
Render( pDC );
|
||
|
|
||
|
ReleaseDC( pDC );
|
||
|
}
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// CGraphControl message handlers
|
||
|
|
||
|
void CGraphControl::OnPaint()
|
||
|
{
|
||
|
CPaintDC dc(this); // device context for painting
|
||
|
|
||
|
Render( &dc );
|
||
|
}
|