|
|
/*++
Copyright (c) 1996 Microsoft Corporation
Module Name:
progbar.c
Abstract:
Centralizes access to the progress bar and associated messages accross components (hwcomp,migapp,etc.) and sides (w9x, nt)
Author:
Marc R. Whitten (marcw) 14-Apr-1997
Revision History:
jimschm 19-Jun-1998 Improved to allow revision of estimates, necessary for NT-side progress bar.
--*/
//
// Includes
//
#include "pch.h"
#define DBG_PROGBAR "Progbar"
//
// Strings
//
// None
//
// Constants
//
#define TICKSCALE 100
//
// Macros
//
// None
//
// Types
//
typedef struct { BOOL Started; BOOL Completed; UINT InitialGuess; UINT TotalTicks; UINT TicksSoFar; UINT LastTickDisplayed; } SLICE, *PSLICE;
typedef struct { HWND Window; HANDLE CancelEvent; PCSTR Message; DWORD MessageId; DWORD Delay; BOOL InUse; } DELAYTHREADPARAMS, *PDELAYTHREADPARAMS;
#if 0
typedef struct { HANDLE CancelEvent; DWORD TickCount; BOOL InUse; } TICKTHREADPARAMS, *PTICKTHREADPARAMS;
#endif
//
// Globals
//
static BOOL g_ProgBarInitialized = FALSE;
static HWND g_ProgressBar; HWND g_Component; HWND g_SubComponent;
static PBRANGE g_OrgRange;
HANDLE g_ComponentCancelEvent; HANDLE g_SubComponentCancelEvent;
static BOOL *g_CancelFlagPtr; static GROWBUFFER g_SliceArray; static UINT g_SliceCount; static UINT g_MaxTickCount; static UINT g_PaddingTicks; static UINT g_CurrentTickCount; static UINT g_CurrentPos; static UINT g_ReduceFactor; static BOOL g_Reverse = FALSE; static OUR_CRITICAL_SECTION g_ProgBarCriticalSection; static UINT g_CurrentSliceId = (UINT)-1; static INT g_ProgBarRefs;
//
// Macro expansion list
//
// None
//
// Private function prototypes
//
// None
//
// Macro expansion definition
//
// None
//
// Code
//
VOID PbInitialize ( IN HWND ProgressBar, IN HWND Component, OPTIONAL IN HWND SubComponent, OPTIONAL IN BOOL *CancelFlagPtr OPTIONAL ) { LONG rc; CHAR Data[256]; DWORD Size; HKEY Key;
MYASSERT (g_ProgBarRefs >= 0);
g_ProgBarRefs++;
if (g_ProgBarRefs == 1) {
g_ProgressBar = ProgressBar; g_CancelFlagPtr = CancelFlagPtr;
g_ProgBarInitialized = TRUE;
SendMessage (ProgressBar, PBM_SETPOS, 0, 0); g_CurrentPos = 0; SendMessage (ProgressBar, PBM_GETRANGE, 0, (LPARAM) &g_OrgRange);
//
// Create cancel events for delayed messages.
//
g_ComponentCancelEvent = CreateEvent (NULL, FALSE, FALSE, NULL); g_SubComponentCancelEvent = CreateEvent (NULL, FALSE, FALSE, NULL);
if (!g_ComponentCancelEvent || !g_SubComponentCancelEvent) { DEBUGMSG ((DBG_ERROR, "ProgressBar: Could not create cancel events.")); }
InitializeOurCriticalSection (&g_ProgBarCriticalSection);
g_Component = Component; g_SubComponent = SubComponent;
DEBUGMSG_IF (( Component && !IsWindow (Component), DBG_WHOOPS, "Progress bar component is not a valid window" ));
DEBUGMSG_IF (( SubComponent && !IsWindow (SubComponent), DBG_WHOOPS, "Progress bar sub component is not a valid window" ));
MYASSERT (!g_SliceCount); MYASSERT (!g_SliceArray.Buf); MYASSERT (!g_MaxTickCount); MYASSERT (!g_PaddingTicks); MYASSERT (!g_CurrentTickCount); MYASSERT (g_CurrentSliceId == (UINT)-1);
g_ReduceFactor = 1;
Key = OpenRegKeyStrA ("HKLM\\inapoi"); if (Key) { Size = 256; rc = RegQueryValueExA (Key, "", NULL, NULL, (PBYTE) Data, &Size); CloseRegKey (Key);
if (rc == ERROR_SUCCESS && !lstrcmpiA (Data, "backwards")) { g_Reverse = TRUE; } } } }
VOID PbTerminate ( VOID ) { MYASSERT (g_ProgBarRefs > 0);
g_ProgBarRefs--;
if (!g_ProgBarRefs) { if (g_ComponentCancelEvent) { CloseHandle (g_ComponentCancelEvent); g_ComponentCancelEvent = NULL; }
if (g_SubComponentCancelEvent) { CloseHandle (g_SubComponentCancelEvent); g_SubComponentCancelEvent = NULL; }
DeleteOurCriticalSection (&g_ProgBarCriticalSection);
GbFree (&g_SliceArray); g_SliceCount = 0; g_MaxTickCount = 0; g_PaddingTicks = 0; g_CurrentTickCount = 0; g_CurrentSliceId = -1; g_Component = NULL; g_SubComponent = NULL;
g_ReduceFactor = 1;
SendMessage (g_ProgressBar, PBM_SETRANGE32, g_OrgRange.iLow, g_OrgRange.iHigh);
g_ProgBarInitialized = FALSE; } }
UINT PbRegisterSlice ( IN UINT InitialEstimate ) { PSLICE Slice; UINT SliceId;
MYASSERT (g_ProgBarInitialized); if (!g_ProgBarInitialized) { return 0; }
SliceId = g_SliceCount;
Slice = (PSLICE) GbGrow (&g_SliceArray, sizeof (SLICE)); g_SliceCount++;
Slice->Started = FALSE; Slice->Completed = FALSE; Slice->TotalTicks = InitialEstimate * TICKSCALE; Slice->InitialGuess = Slice->TotalTicks; Slice->TicksSoFar = 0; Slice->LastTickDisplayed = 0;
return SliceId; }
VOID PbReviseSliceEstimate ( IN UINT SliceId, IN UINT RevisedEstimate ) { PSLICE Slice;
MYASSERT (g_ProgBarInitialized); if (!g_ProgBarInitialized) { return; }
if (SliceId >= g_SliceCount) { DEBUGMSG ((DBG_WHOOPS, "ReviseSliceEstimate: Invalid slice ID %u", SliceId)); return; }
Slice = (PSLICE) g_SliceArray.Buf + SliceId;
if (!g_CurrentTickCount) { Slice->TotalTicks = RevisedEstimate; return; }
if (Slice->Completed) { DEBUGMSG ((DBG_WHOOPS, "ReviseSliceEstimate: Can't revise completed slice")); return; }
if (Slice->InitialGuess == 0) { return; }
RevisedEstimate *= TICKSCALE;
MYASSERT (Slice->TicksSoFar * RevisedEstimate >= Slice->TicksSoFar); MYASSERT (Slice->LastTickDisplayed * RevisedEstimate >= Slice->LastTickDisplayed);
Slice->TicksSoFar = (UINT) ((LONGLONG) Slice->TicksSoFar * (LONGLONG) RevisedEstimate / (LONGLONG) Slice->TotalTicks); Slice->LastTickDisplayed = (UINT) ((LONGLONG) Slice->LastTickDisplayed * (LONGLONG) RevisedEstimate / (LONGLONG) Slice->TotalTicks); Slice->TotalTicks = RevisedEstimate; }
VOID PbBeginSliceProcessing ( IN UINT SliceId ) { PSLICE Slice; UINT u; UINT TotalTicks;
MYASSERT (g_ProgBarInitialized); if (!g_ProgBarInitialized) { return; }
if (!g_ProgressBar) { DEBUGMSG ((DBG_WHOOPS, "No progress bar handle")); return; }
if (SliceId >= g_SliceCount) { DEBUGMSG ((DBG_WHOOPS, "BeginSliceProcessing: Invalid slice ID %u", SliceId)); return; }
if (!g_CurrentTickCount) { //
// Initialize the progress bar
//
MYASSERT (g_CurrentSliceId == (UINT)-1);
TotalTicks = 0; Slice = (PSLICE) g_SliceArray.Buf;
for (u = 0 ; u < g_SliceCount ; u++) { TotalTicks += Slice->InitialGuess; Slice++; }
TotalTicks /= TICKSCALE; g_PaddingTicks = TotalTicks * 5 / 100; g_MaxTickCount = TotalTicks + 2 * g_PaddingTicks;
g_ReduceFactor = 1; while (g_MaxTickCount > 0xffff) { g_ReduceFactor *= 10; g_MaxTickCount /= 10; }
SendMessage (g_ProgressBar, PBM_SETRANGE, 0, MAKELPARAM (0, g_MaxTickCount)); SendMessage (g_ProgressBar, PBM_SETSTEP, 1, 0);
if (g_Reverse) { SendMessage ( g_ProgressBar, PBM_SETPOS, g_MaxTickCount - (g_PaddingTicks / g_ReduceFactor), 0 ); } else { SendMessage (g_ProgressBar, PBM_SETPOS, g_PaddingTicks / g_ReduceFactor, 0); }
g_CurrentTickCount = g_PaddingTicks; g_CurrentPos = g_PaddingTicks;
} else if (SliceId <= g_CurrentSliceId) { DEBUGMSG ((DBG_WHOOPS, "BeginSliceProcessing: Slice ID %u processed already", SliceId)); return; }
g_CurrentSliceId = SliceId; Slice = (PSLICE) g_SliceArray.Buf + g_CurrentSliceId;
Slice->Started = TRUE; }
VOID pIncrementBarIfNecessary ( IN OUT PSLICE Slice ) { UINT Increment; UINT Pos;
if (Slice->TicksSoFar >= Slice->TotalTicks) { Slice->TicksSoFar = Slice->TotalTicks; Slice->Completed = TRUE; }
if (Slice->TicksSoFar - Slice->LastTickDisplayed >= TICKSCALE) { Increment = (Slice->TicksSoFar - Slice->LastTickDisplayed) / TICKSCALE; Slice->LastTickDisplayed += Increment * TICKSCALE;
Pos = ((g_CurrentPos + Slice->TicksSoFar) / TICKSCALE);
Pos += g_PaddingTicks; Pos /= g_ReduceFactor;
if (Pos > g_MaxTickCount) { Pos = g_MaxTickCount - (g_PaddingTicks / g_ReduceFactor); }
if (g_Reverse) {
SendMessage (g_ProgressBar, PBM_SETPOS, g_MaxTickCount - Pos, 0);
} else {
SendMessage (g_ProgressBar, PBM_SETPOS, Pos, 0); } } }
VOID static pTickProgressBar ( IN UINT Ticks ) { PSLICE Slice; LONGLONG x;
if (!Ticks || g_CurrentSliceId == (UINT)-1 || g_CurrentSliceId >= g_SliceCount) { return; }
Slice = (PSLICE) g_SliceArray.Buf + g_CurrentSliceId;
if (!Slice->InitialGuess) { return; }
if (Slice->Completed) { DEBUGMSG ((DBG_WARNING, "Slice ID %u already completed", g_CurrentSliceId)); return; }
MYASSERT (Ticks * TICKSCALE > Ticks); x = ((LONGLONG) Ticks * TICKSCALE * (LONGLONG) Slice->TotalTicks) / (LONGLONG) Slice->InitialGuess; MYASSERT (x + (LONGLONG) Slice->TicksSoFar < 0x100000000);
Slice->TicksSoFar += (UINT) x;
pIncrementBarIfNecessary (Slice);
}
BOOL PbTickDelta ( IN UINT TickCount ) { BOOL rSuccess = TRUE;
MYASSERT (g_ProgBarInitialized); if (!g_ProgBarInitialized) { return TRUE; }
if (g_CancelFlagPtr && *g_CancelFlagPtr) { SetLastError (ERROR_CANCELLED); rSuccess = FALSE; } else { pTickProgressBar (TickCount); }
return rSuccess; }
BOOL PbTick ( VOID ) { MYASSERT (g_ProgBarInitialized); if (!g_ProgBarInitialized) { return TRUE; }
return PbTickDelta (1); }
VOID PbGetSliceInfo ( IN UINT SliceId, OUT PBOOL SliceStarted, OPTIONAL OUT PBOOL SliceFinished, OPTIONAL OUT PUINT TicksCompleted, OPTIONAL OUT PUINT TotalTicks OPTIONAL ) { PSLICE Slice;
Slice = (PSLICE) g_SliceArray.Buf + SliceId;
if (SliceStarted) { *SliceStarted = Slice->Started; }
if (SliceFinished) { *SliceFinished = Slice->Completed; }
if (TicksCompleted) { *TicksCompleted = Slice->TicksSoFar / TICKSCALE; }
if (TotalTicks) { *TotalTicks = Slice->TotalTicks / TICKSCALE; } }
VOID PbEndSliceProcessing ( VOID ) { PSLICE Slice;
MYASSERT (g_ProgBarInitialized); if (!g_ProgBarInitialized) { return; }
Slice = (PSLICE) g_SliceArray.Buf + g_CurrentSliceId;
if (!Slice->InitialGuess) { Slice->Completed = TRUE; return; }
if (!Slice->Completed) { DEBUGMSG ((DBG_WARNING, "Progress bar slice %u was not completed.", g_CurrentSliceId));
Slice->TicksSoFar = Slice->TotalTicks; Slice->Completed = TRUE;
pIncrementBarIfNecessary (Slice); }
g_CurrentPos += Slice->TotalTicks;
if (g_CurrentSliceId == g_SliceCount - 1) { //
// End of progress bar
//
SendMessage (g_ProgressBar, PBM_SETPOS, g_MaxTickCount, 0); } }
BOOL pCheckProgressBarState ( IN HANDLE CancelEvent ) {
SetEvent(CancelEvent);
return (!g_CancelFlagPtr || !*g_CancelFlagPtr); }
BOOL PbSetWindowStringA ( IN HWND Window, IN HANDLE CancelEvent, IN PCSTR Message, OPTIONAL IN DWORD MessageId OPTIONAL ) { BOOL rSuccess = TRUE; PCSTR string = NULL;
EnterOurCriticalSection (&g_ProgBarCriticalSection);
if (g_ProgBarInitialized) {
if (pCheckProgressBarState(CancelEvent)) {
if (Message) {
//
// We have a normal message string.
//
if (!SetWindowTextA(Window, Message)) { rSuccess = FALSE; DEBUGMSG((DBG_ERROR,"ProgressBar: SetWindowText failed.")); } } else if (MessageId) {
//
// We have a message ID. Convert it and set it.
//
string = GetStringResourceA(MessageId);
if (string) {
if (!SetWindowTextA(Window, string)) { rSuccess = FALSE; DEBUGMSG((DBG_ERROR,"ProgressBar: SetWindowText failed.")); }
FreeStringResourceA(string); } ELSE_DEBUGMSG((DBG_ERROR,"ProgressBar: Error with GetStringResource"));
} else {
//
// Just clear the text.
//
if (!SetWindowTextA(Window, "")) { rSuccess = FALSE; DEBUGMSG((DBG_ERROR,"ProgressBar: SetWindowText failed.")); } } } else { //
// We are in a canceled state.
//
rSuccess = FALSE; SetLastError (ERROR_CANCELLED); } }
LeaveOurCriticalSection (&g_ProgBarCriticalSection);
return rSuccess;
}
DWORD pSetDelayedMessageA ( IN PVOID Param ) { DWORD rc = ERROR_SUCCESS; PDELAYTHREADPARAMS tParams = (PDELAYTHREADPARAMS) Param;
//
// Simply wait for the passed in delay or until someone signals the cancel
// event.
//
switch (WaitForSingleObject(tParams -> CancelEvent, tParams -> Delay)) {
case WAIT_TIMEOUT: //
// We timed out without cancel being signaled. Set the delayed message.
//
PbSetWindowStringA ( tParams->Window, tParams->CancelEvent, tParams->Message, tParams->MessageId );
break;
case WAIT_OBJECT_0: default: //
// We were canceled (or something strange happened :> Do nothing!
//
break; }
//
// can set a new thread now
//
tParams->InUse = FALSE;
return rc; }
VOID PbCancelDelayedMessage ( IN HANDLE CancelEvent ) { if (!g_ProgBarInitialized) { return; }
SetEvent(CancelEvent);
}
BOOL PbSetDelayedMessageA ( IN HWND Window, IN HANDLE CancelEvent, IN LPCSTR Message, IN DWORD MessageId, IN DWORD Delay ) { BOOL rSuccess = FALSE; DWORD threadId; static DELAYTHREADPARAMS tParams;
if (!g_ProgBarInitialized || tParams.InUse) { return TRUE; }
if (!pCheckProgressBarState(Window)) {
//
// Fill in the parameters for this call to create thread.
//
tParams.Window = Window; tParams.CancelEvent = CancelEvent; tParams.Message = Message; tParams.MessageId = MessageId; tParams.Delay = Delay;
//
// Spawn off a thread that will set the message.
//
rSuccess = NULL != CreateThread ( NULL, // No inheritance.
0, // Normal stack size.
pSetDelayedMessageA, &tParams, 0, // Run immediately.
&threadId );
if (rSuccess) { tParams.InUse = TRUE; } ELSE_DEBUGMSG((DBG_ERROR,"Error spawning thread in PbSetDelayedMessageA.")); }
return rSuccess; }
#if 0
DWORD pTickProgressBarThread ( IN PVOID Param ) { DWORD rc = ERROR_SUCCESS; PTICKTHREADPARAMS Params = (PTICKTHREADPARAMS)Param; BOOL Continue = TRUE;
//
// Simply wait for the passed in delay or until someone signals the cancel
// event.
//
do { switch (WaitForSingleObject(Params->CancelEvent, Params->TickCount)) {
case WAIT_TIMEOUT: //
// We timed out without cancel being signaled. Tick the progress bar.
//
if (!PbTickDelta (Params->TickCount)) { //
// cancelled
//
Continue = FALSE; } break;
case WAIT_OBJECT_0: default: //
// We were canceled (or something strange happened :> Do nothing!
//
Continue = FALSE; break; } } while (Continue);
//
// can set a new thread now
//
Params->InUse = FALSE;
return rc; }
BOOL PbCreateTickThread ( IN HANDLE CancelEvent, IN DWORD TickCount ) { BOOL rSuccess = FALSE; DWORD threadId; static TICKTHREADPARAMS g_Params;
if (g_ProgBarInitialized && !g_Params.InUse) {
if (pCheckProgressBarState(NULL)) {
//
// Fill in the parameters for this call to create thread.
//
g_Params.CancelEvent = CancelEvent; g_Params.TickCount = TickCount;
//
// Spawn off a thread that will set the message.
//
if (CreateThread ( NULL, // No inheritance.
0, // Normal stack size.
pTickProgressBarThread, &g_Params, 0, // Run immediately.
&threadId )) { rSuccess = TRUE; g_Params.InUse = TRUE; } ELSE_DEBUGMSG ((DBG_ERROR, "Error spawning thread in PbCreateTickThread.")); } }
return rSuccess; }
BOOL PbCancelTickThread ( IN HANDLE CancelEvent ) { if (!g_ProgBarInitialized) { return TRUE; }
return SetEvent(CancelEvent); }
#endif
|