Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

884 lines
18 KiB

/*++
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