2020-09-30 16:53:55 +02:00

516 lines
13 KiB
C++

/*++
Copyright (C) 1996-1999 Microsoft Corporation
Module Name:
timerng.cpp
Abstract:
<abstract>
--*/
//==========================================================================//
// Includes //
//==========================================================================//
#include <assert.h>
#include <windows.h>
#include "winhelpr.h"
#include "utils.h"
#include "timerng.h"
#include "globals.h"
#define INTRVBAR_RANGE 1000
#define MAX_TIME_CHARS 20
#define MAX_DATE_CHARS 20
WCHAR szTimeRangeClass[] = L"TimeRange";
INT
CTimeRange::MaxTimeWidth (
HDC hDC
)
/*
Effect: Return a reasonable maximum number of pixels to hold
expected time and date strings.
To Do: When we use the alleged local-date and local-time display
functions, we will modify this routine to use them.
*/
{
return (max (TextWidth (hDC, L" 99 XXX 99 "),
TextWidth (hDC, L" 99:99:99.9 PM "))) ;
}
void
CTimeRange::DrawBeginEnd (
HDC hDC
)
{
WCHAR szDate [MAX_TIME_CHARS] ;
WCHAR szTime [MAX_DATE_CHARS] ;
SetTextAlign (hDC, TA_TOP) ;
SelectFont (hDC, m_hFont) ;
SetBkColor(hDC, GetSysColor(COLOR_3DFACE));
SetTextColor(hDC, GetSysColor(COLOR_BTNTEXT));
// Draw the begin time
FormatDateTime(m_llBegin, szTime, szDate);
SetTextAlign (hDC, TA_RIGHT) ;
TextOut (hDC, m_xBegin, 0, szDate, lstrlen (szDate)) ;
TextOut (hDC, m_xBegin, m_yFontHeight, szTime, lstrlen (szTime)) ;
// Draw the end time
FormatDateTime(m_llEnd, szTime, szDate);
SetTextAlign (hDC, TA_LEFT) ;
TextOut (hDC, m_xEnd, 0, szDate, lstrlen (szDate)) ;
TextOut (hDC, m_xEnd, m_yFontHeight, szTime, lstrlen (szTime)) ;
}
void
CTimeRange::DrawStartStop (
HDC hDC
)
/*
Effect: Draw the start and stop date/times on the bottom of the
timeline. Draw the start date/time right justified at the
outer edge of the start point and the stop date/time left
justified with the outer edge of the stop point.
Erase previous start and stop date/times in the process.
*/
{
RECT rectDate ;
RECT rectTime ;
RECT rectOpaque ;
WCHAR szTime [MAX_TIME_CHARS] ;
WCHAR szDate [MAX_DATE_CHARS] ;
INT xStart ;
INT xStop ;
INT iStart ;
INT iStop ;
INT xDateTimeWidth ;
SelectFont (hDC, m_hFont) ;
SetTextAlign (hDC, TA_TOP) ;
SetBkColor(hDC, GetSysColor(COLOR_3DFACE));
SetTextColor(hDC, GetSysColor(COLOR_BTNTEXT));
//=============================//
// Get Start Information //
//=============================//
xStart = m_xBegin + m_pIntrvBar->XStart();
iStart = m_pIntrvBar->Start() ;
m_llStart = m_llBegin + ((m_llEnd - m_llBegin) * iStart) / INTRVBAR_RANGE;
FormatDateTime(m_llStart, szTime, szDate);
xDateTimeWidth = max (TextWidth (hDC, szDate),
TextWidth (hDC, szTime)) ;
//=============================//
// Write Start Date //
//=============================//
rectDate.left = xStart - xDateTimeWidth ;
rectDate.top = m_rectStartDate.top ;
rectDate.right = xStart ;
rectDate.bottom = m_rectStartDate.bottom ;
SetTextAlign (hDC, TA_RIGHT) ;
UnionRect (&rectOpaque, &m_rectStartDate, &rectDate) ;
ExtTextOut (hDC,
rectDate.right, rectDate.top,
ETO_OPAQUE,
&rectOpaque,
szDate, lstrlen (szDate),
NULL) ;
m_rectStartDate = rectDate ;
//=============================//
// Write Start Time //
//=============================//
rectTime.left = rectDate.left ;
rectTime.top = m_rectStartTime.top ;
rectTime.right = rectDate.right ;
rectTime.bottom = m_rectStartTime.bottom ;
UnionRect (&rectOpaque, &m_rectStartTime, &rectTime) ;
ExtTextOut (hDC,
rectTime.right, rectTime.top,
ETO_OPAQUE,
&rectOpaque,
szTime, lstrlen (szTime),
NULL) ;
m_rectStartTime = rectTime ;
//=============================//
// Get Stop Information //
//=============================//
xStop = m_xBegin + m_pIntrvBar->XStop() ;
iStop = m_pIntrvBar->Stop () ;
m_llStop = m_llBegin + ((m_llEnd - m_llBegin) * iStop) / INTRVBAR_RANGE;
FormatDateTime(m_llStop, szTime, szDate);
xDateTimeWidth = max (TextWidth (hDC, szDate),
TextWidth (hDC, szTime)) ;
//=============================//
// Write Stop Date //
//=============================//
rectDate.left = xStop ;
rectDate.top = m_rectStopDate.top ;
rectDate.right = xStop + xDateTimeWidth ;
rectDate.bottom = m_rectStopDate.bottom ;
SetTextAlign (hDC, TA_LEFT) ;
UnionRect (&rectOpaque, &m_rectStopDate, &rectDate) ;
ExtTextOut (hDC,
rectDate.left, rectDate.top,
ETO_OPAQUE,
&rectOpaque,
szDate, lstrlen (szDate),
NULL) ;
m_rectStopDate = rectDate ;
//=============================//
// Write Stop Time //
//=============================//
rectTime.left = rectDate.left ;
rectTime.top = m_rectStopTime.top ;
rectTime.right = rectDate.right ;
rectTime.bottom = m_rectStopTime.bottom ;
UnionRect (&rectOpaque, &m_rectStopTime, &rectTime) ;
ExtTextOut (hDC,
rectTime.left, rectTime.top,
ETO_OPAQUE,
&rectOpaque,
szTime, lstrlen (szTime),
NULL) ;
m_rectStopTime = rectTime ;
}
//==========================================================================//
// Message Handlers //
//==========================================================================//
CTimeRange::CTimeRange (
HWND hWnd )
{
HDC hDC ;
// Caller checks for NULL hwnd before calling this procedure
assert ( NULL != hWnd );
m_hWnd = hWnd ;
SetWindowLongPtr(hWnd, 0, (INT_PTR)this);
m_hFont = (HFONT)GetStockObject(DEFAULT_GUI_FONT) ;
hDC = GetDC (hWnd) ;
if (hDC) {
if ( NULL != m_hFont ) {
SelectFont (hDC, m_hFont) ;
}
m_yFontHeight = FontHeight (hDC, TRUE) ;
m_xMaxTimeWidth = MaxTimeWidth (hDC) ;
ReleaseDC (hWnd, hDC) ;
} else {
m_yFontHeight = 0;
m_xMaxTimeWidth = 0;
}
m_pIntrvBar = NULL;
}
CTimeRange::~CTimeRange (
void
)
{
if (m_pIntrvBar != NULL) {
delete m_pIntrvBar;
m_pIntrvBar = NULL;
}
}
BOOL
CTimeRange::Init(
void
)
{
RECT rc;
m_pIntrvBar = new CIntervalBar();
if (m_pIntrvBar != NULL) {
if (m_pIntrvBar->Init(m_hWnd)) {
// Size now because we didn't exist when the window
// got the initial WM_SIZE message
GetWindowRect(m_hWnd, &rc);
OnSize(rc.right - rc.left, rc.bottom - rc.top);
m_pIntrvBar->SetRange(0, INTRVBAR_RANGE);
}
else {
delete m_pIntrvBar;
m_pIntrvBar = NULL;
}
}
return ( NULL != m_pIntrvBar );
}
void
CTimeRange::OnSize (
INT xWidth,
INT yHeight
)
/*
Effect: Perform all actions needed when the size of the timeline
hWnd has changed. In particular, determine the appropriate
size for the ILine window and set the rectangles for the
top and bottom displays.
*/
{
INT yLine ;
INT yDate, yTime ;
INT xEnd ;
xEnd = xWidth - m_xMaxTimeWidth ;
yLine = m_yFontHeight ;
yDate = yHeight - 2 * yLine ;
yTime = yHeight - yLine ;
SetRect (&m_rectStartDate, 0, yDate, 0, yDate + yLine) ;
SetRect (&m_rectStartTime, 0, yTime, 0, yTime + yLine) ;
SetRect (&m_rectStopDate, xEnd, yDate, xEnd, yDate + yLine) ;
SetRect (&m_rectStopTime, xEnd, yTime, xEnd, yTime + yLine) ;
MoveWindow (m_pIntrvBar->Window(),
m_xMaxTimeWidth, 2 * m_yFontHeight,
xWidth - 2 * m_xMaxTimeWidth,
yHeight - 4 * m_yFontHeight,
FALSE) ;
m_xBegin = m_xMaxTimeWidth ;
m_xEnd = xWidth - m_xMaxTimeWidth ;
}
//==========================================================================//
// Exported Functions //
//==========================================================================//
LRESULT APIENTRY TimeRangeWndProc (
HWND hWnd,
UINT uiMsg,
WPARAM wParam,
LPARAM lParam
)
/*
Note: This function must be declared in the application's
linker-definition file, perfmon.def file.
*/
{
BOOL bCallDefWindowProc ;
LRESULT lrsltReturnValue ;
PCTimeRange pTimeRange = (PCTimeRange) GetWindowLongPtr(hWnd, 0);;
bCallDefWindowProc = FALSE ;
lrsltReturnValue = 0L ;
if (pTimeRange) {
switch ( uiMsg ) {
case WM_SETFOCUS:
return 0 ;
case WM_KILLFOCUS:
return 0 ;
case WM_ENABLE:
WindowInvalidate(hWnd);
EnableWindow(pTimeRange->m_pIntrvBar->Window(), (BOOL)wParam);
break;
case WM_COMMAND:
{
HDC hDC;
hDC = GetDC (hWnd) ;
if (hDC) {
pTimeRange->DrawStartStop (hDC) ;
ReleaseDC (hWnd, hDC) ;
}
SendMessage(
WindowParent(hWnd),
WM_COMMAND, (WPARAM)WindowID(hWnd), (LPARAM)hWnd);
}
break ;
case WM_PAINT:
{
HDC hDC ;
PAINTSTRUCT ps ;
hDC = BeginPaint (hWnd, &ps) ;
if (hDC) {
if (IsWindowEnabled(hWnd)) {
pTimeRange->DrawBeginEnd (hDC) ;
pTimeRange->DrawStartStop (hDC) ;
}
EndPaint (hWnd, &ps) ;
}
}
break ;
case WM_SIZE:
pTimeRange->OnSize (LOWORD (lParam), HIWORD (lParam)) ;
break ;
default:
bCallDefWindowProc = TRUE ;
}
}
else {
bCallDefWindowProc = TRUE;
}
if (bCallDefWindowProc)
lrsltReturnValue = DefWindowProc (hWnd, uiMsg, wParam, lParam) ;
return (lrsltReturnValue) ;
}
BOOL RegisterTimeRangeClass (
void
)
{
#define dwTimeRangeClassStyle (CS_HREDRAW | CS_VREDRAW)
BEGIN_CRITICAL_SECTION
// Register window class once
if (pstrRegisteredClasses[TIMERANGE_WNDCLASS] == NULL) {
WNDCLASS wc ;
wc.style = dwTimeRangeClassStyle ;
wc.lpfnWndProc = TimeRangeWndProc ;
wc.cbClsExtra = 0 ;
wc.cbWndExtra = sizeof(PCTimeRange);
wc.hInstance = g_hInstance ;
wc.hIcon = NULL ;
wc.hCursor = LoadCursor (NULL, IDC_ARROW) ;
wc.hbrBackground = (HBRUSH) (COLOR_3DFACE + 1) ;
wc.lpszMenuName = NULL ;
wc.lpszClassName = szTimeRangeClass ;
if (RegisterClass (&wc)) {
pstrRegisteredClasses[TIMERANGE_WNDCLASS] = szTimeRangeClass;
}
}
END_CRITICAL_SECTION
return (pstrRegisteredClasses[LEGEND_WNDCLASS] != NULL);
}
void
CTimeRange::SetBeginEnd (
LONGLONG llBegin,
LONGLONG llEnd
)
{
HDC hDC;
m_llBegin = llBegin;
m_llEnd = llEnd;
hDC = GetDC (m_hWnd) ;
if (hDC) {
DrawBeginEnd (hDC) ;
ReleaseDC (m_hWnd, hDC) ;
}
}
void
CTimeRange::SetStartStop (
LONGLONG llStart,
LONGLONG llStop
)
{
INT iStart;
INT iStop;
HDC hDC;
LONGLONG llBeginToEnd;
if (llStart < m_llBegin)
llStart = m_llBegin;
if (llStop > m_llEnd)
llStop = m_llEnd;
m_llStart = llStart;
m_llStop = llStop;
llBeginToEnd = m_llEnd - m_llBegin;
if ( 0 != llBeginToEnd ) {
iStart = (INT)(((llStart - m_llBegin) * INTRVBAR_RANGE) / llBeginToEnd);
iStop = (INT)(((llStop - m_llBegin) * INTRVBAR_RANGE) / llBeginToEnd);
} else {
iStart = 0;
iStop = 0;
}
m_pIntrvBar->SetStart(iStart);
m_pIntrvBar->SetStop(iStop);
hDC = GetDC (m_hWnd) ;
if (hDC) {
DrawStartStop (hDC) ;
ReleaseDC (m_hWnd, hDC) ;
}
}