//========= 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 );
}