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.
 
 
 
 
 
 

630 lines
14 KiB

// ProgressBar.cpp : implementation file
/////////////////////////////////////////////////////////////////////////////
#include "stdafx.h"
#include "ProgressBar.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
IMPLEMENT_DYNCREATE(CProgressBar, CProgressCtrl)
BEGIN_MESSAGE_MAP(CProgressBar, CProgressCtrl)
//{{AFX_MSG_MAP(CProgressBar)
ON_WM_ERASEBKGND()
//}}AFX_MSG_MAP
END_MESSAGE_MAP()
/**************************************************************************\
* CProgressBar::CProgressBar()
*
* Constructor for the progress control
*
*
* Arguments:
*
* none
*
* Return Value:
*
* none
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
CProgressBar::CProgressBar()
{
m_Rect.SetRect(0,0,0,0);
}
/**************************************************************************\
* CProgressBar::
*
* Constructor for the Progress control
*
*
* Arguments:
*
* strMessage - Status message to display
* nSize - Max range of Progress control
* MaxValue - Max range of Progress control
* bSmooth - Smooth/Normal mode of progress display
* nPane - which pane to display the progress control
*
* Return Value:
*
* none
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
CProgressBar::CProgressBar(LPCTSTR strMessage, int nSize /*=100*/,
int MaxValue /*=100*/, BOOL bSmooth /*=FALSE*/,
int nPane/*=0*/)
{
Create(strMessage, nSize, MaxValue, bSmooth, nPane);
}
/**************************************************************************\
* CProgressBar::~CProgressBar()
*
* Destruction
*
*
* Arguments:
*
* none
*
* Return Value:
*
* none
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
CProgressBar::~CProgressBar()
{
Clear();
}
/**************************************************************************\
* CProgressBar::GetStatusBar()
*
* Returns the Status bar
*
*
* Arguments:
*
* none
*
* Return Value:
*
* CStatusBar* - Status bar to be returned
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
CStatusBar* CProgressBar::GetStatusBar()
{
CWnd *pMainWnd = AfxGetMainWnd();
if (!pMainWnd)
return NULL;
// If main window is a frame window, use normal methods...
if (pMainWnd->IsKindOf(RUNTIME_CLASS(CFrameWnd)))
{
CWnd* pMessageBar = ((CFrameWnd*)pMainWnd)->GetMessageBar();
return DYNAMIC_DOWNCAST(CStatusBar, pMessageBar);
}
// otherwise traverse children to try and find the status bar...
else
return DYNAMIC_DOWNCAST(CStatusBar,
pMainWnd->GetDescendantWindow(AFX_IDW_STATUS_BAR));
}
/**************************************************************************\
* CProgressBar::Create()
*
* Creates a new progress control
* Create the CProgressCtrl as a child of the status bar positioned
* over the first pane, extending "nSize" percentage across pane.
* Sets the range to be 0 to MaxValue, with a step of 1.
*
*
* Arguments:
*
* strMessage - Status message to display
* nSize - Max range of Progress control
* MaxValue - Max range of Progress control
* bSmooth - Smooth/Normal mode of progress display
* nPane - which pane to display the progress control
*
* Return Value:
*
* none
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
BOOL CProgressBar::Create(LPCTSTR strMessage, int nSize /*=100*/,
int MaxValue /*=100*/, BOOL bSmooth /*=FALSE*/, int nPane/*=0*/)
{
BOOL bSuccess = FALSE;
CStatusBar *pStatusBar = GetStatusBar();
if (!pStatusBar)
return FALSE;
DWORD dwStyle = WS_CHILD|WS_VISIBLE;
#ifdef PBS_SMOOTH
if (bSmooth)
dwStyle |= PBS_SMOOTH;
#endif
// Get CRect coordinates for requested status bar pane
CRect PaneRect;
pStatusBar->GetItemRect(nPane, &PaneRect);
// Create the progress bar
bSuccess = CProgressCtrl::Create(dwStyle, PaneRect, pStatusBar, 1);
ASSERT(bSuccess);
if (!bSuccess)
return FALSE;
// Set range and step
SetRange(0, MaxValue);
SetStep(1);
m_strMessage = strMessage;
m_nSize = nSize;
m_nPane = nPane;
m_strPrevText = pStatusBar->GetPaneText(m_nPane);
// Resize the control to its desired width
Resize();
return TRUE;
}
/**************************************************************************\
* CProgressBar::Clear()
*
* Destroy the progress control
*
*
* Arguments:
*
* none
*
* Return Value:
*
* none
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
void CProgressBar::Clear()
{
if (!IsWindow(GetSafeHwnd()))
return;
// Hide the window. This is necessary so that a cleared
// window is not redrawn if "Resize" is called
ModifyStyle(WS_VISIBLE, 0);
CString str;
if (m_nPane == 0)
str.LoadString(AFX_IDS_IDLEMESSAGE); // Get the IDLE_MESSAGE
else
str = m_strPrevText; // Restore previous text
// Place the IDLE_MESSAGE in the status bar
CStatusBar *pStatusBar = GetStatusBar();
if (pStatusBar)
{
pStatusBar->SetPaneText(m_nPane, str);
pStatusBar->UpdateWindow();
}
}
/**************************************************************************\
* CProgressBar::SetText()
*
* Set the display text for the progress control
*
*
* Arguments:
*
* none
*
* Return Value:
*
* none
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
BOOL CProgressBar::SetText(LPCTSTR strMessage)
{
m_strMessage = strMessage;
SetPaneText(m_nPane,m_strMessage);
return Resize();
}
/**************************************************************************\
* CProgressBar::SetSize()
*
* Set the size for the progress control
*
*
* Arguments:
*
* nSize - New size for the progress control
*
* Return Value:
*
* none
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
BOOL CProgressBar::SetSize(int nSize)
{
m_nSize = nSize;
return Resize();
}
/**************************************************************************\
* CProgressBar::SetBarColour()
*
* Sets the color for the progress control
*
*
* Arguments:
*
* clrBar - Color for progress control
*
* Return Value:
*
* COLORREF last color of progress control
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
COLORREF CProgressBar::SetBarColour(COLORREF clrBar)
{
#ifdef PBM_SETBKCOLOR
if (!IsWindow(GetSafeHwnd()))
return CLR_DEFAULT;
return (COLORREF )SendMessage(PBM_SETBARCOLOR, 0, (LPARAM) clrBar);
#else
UNUSED(clrBar);
return CLR_DEFAULT;
#endif
}
/**************************************************************************\
* CProgressBar::SetBkColour()
*
* Sets the background color for the progress control
*
*
* Arguments:
*
* none
*
* Return Value:
*
* COLORREF last color of progress control background
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
COLORREF CProgressBar::SetBkColour(COLORREF clrBk)
{
#ifdef PBM_SETBKCOLOR
if (!IsWindow(GetSafeHwnd()))
return CLR_DEFAULT;
return (COLORREF) SendMessage(PBM_SETBKCOLOR, 0, (LPARAM) clrBk);
#else
UNUSED(clrBk);
return CLR_DEFAULT;
#endif
}
/**************************************************************************\
* CProgressBar::SetRange()
*
* Set the range limits for the progress control
*
*
* Arguments:
*
* nLower - Lower limit
* nUpper - Upper limit
* nStep - Step value
*
* Return Value:
*
* status
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
BOOL CProgressBar::SetRange(int nLower, int nUpper, int nStep /* = 1 */)
{
if (!IsWindow(GetSafeHwnd()))
return FALSE;
// To take advantage of the Extended Range Values we use the PBM_SETRANGE32
// message intead of calling CProgressCtrl::SetRange directly. If this is
// being compiled under something less than VC 5.0, the necessary defines
// may not be available.
#ifdef PBM_SETRANGE32
ASSERT(-0x7FFFFFFF <= nLower && nLower <= 0x7FFFFFFF);
ASSERT(-0x7FFFFFFF <= nUpper && nUpper <= 0x7FFFFFFF);
SendMessage(PBM_SETRANGE32, (WPARAM) nLower, (LPARAM) nUpper);
#else
ASSERT(0 <= nLower && nLower <= 65535);
ASSERT(0 <= nUpper && nUpper <= 65535);
CProgressCtrl::SetRange(nLower, nUpper);
#endif
CProgressCtrl::SetStep(nStep);
return TRUE;
}
/**************************************************************************\
* CProgressBar::SetPos()
*
* Sets the current position for the progress control
*
*
* Arguments:
*
* nPos - new Postion to be set
*
* Return Value:
*
* status
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
int CProgressBar::SetPos(int nPos)
{
if (!IsWindow(GetSafeHwnd()))
return 0;
#ifdef PBM_SETRANGE32
ASSERT(-0x7FFFFFFF <= nPos && nPos <= 0x7FFFFFFF);
#else
ASSERT(0 <= nPos && nPos <= 65535);
#endif
ModifyStyle(0,WS_VISIBLE);
return CProgressCtrl::SetPos(nPos);
}
/**************************************************************************\
* CProgressBar::OffestPos()
*
* Set the progress control's offset
*
*
* Arguments:
*
* nPos - Position offset
*
* Return Value:
*
* status
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
int CProgressBar::OffsetPos(int nPos)
{
if (!IsWindow(GetSafeHwnd()))
return 0;
ModifyStyle(0,WS_VISIBLE);
return CProgressCtrl::OffsetPos(nPos);
}
/**************************************************************************\
* CProgressBar::SetStep()
*
* Set progress control's step value
*
*
* Arguments:
*
* nStep - Step value to be set
*
* Return Value:
*
* none
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
int CProgressBar::SetStep(int nStep)
{
if (!IsWindow(GetSafeHwnd()))
return 0;
ModifyStyle(0,WS_VISIBLE);
return CProgressCtrl::SetStep(nStep);
}
/**************************************************************************\
* CProgressBar::StepIt()
*
* Step the progress control standard step value
*
*
* Arguments:
*
* none
*
* Return Value:
*
* none
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
int CProgressBar::StepIt()
{
if (!IsWindow(GetSafeHwnd()))
return 0;
ModifyStyle(0,WS_VISIBLE);
return CProgressCtrl::StepIt();
}
/**************************************************************************\
* CProgressBar::Resize()
*
* Resize the progress control to fit
*
*
* Arguments:
*
* none
*
* Return Value:
*
* status
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
BOOL CProgressBar::Resize()
{
if (!IsWindow(GetSafeHwnd()))
return FALSE;
CStatusBar *pStatusBar = GetStatusBar();
if (!pStatusBar)
return FALSE;
// Redraw the window text
if (IsWindowVisible())
{
pStatusBar->SetPaneText(m_nPane, m_strMessage);
pStatusBar->UpdateWindow();
}
// Calculate how much space the text takes up
CClientDC dc(pStatusBar);
CFont *pOldFont = dc.SelectObject(pStatusBar->GetFont());
CSize size = dc.GetTextExtent(m_strMessage); // Length of text
int margin = dc.GetTextExtent(_T(" ")).cx * 2; // Text margin
dc.SelectObject(pOldFont);
// Now calculate the rectangle in which we will draw the progress bar
CRect rc;
pStatusBar->GetItemRect(m_nPane, rc);
// Position left of progress bar after text and right of progress bar
// to requested percentage of status bar pane
if (!m_strMessage.IsEmpty())
rc.left += (size.cx + 2*margin);
rc.right -= (rc.right - rc.left) * (100 - m_nSize) / 100;
if (rc.right < rc.left) rc.right = rc.left;
// Leave a litle vertical margin (10%) between the top and bottom of the bar
int Height = rc.bottom - rc.top;
rc.bottom -= Height/10;
rc.top += Height/10;
// If the window size has changed, resize the window
if (rc != m_Rect)
{
MoveWindow(&rc);
m_Rect = rc;
}
return TRUE;
}
/**************************************************************************\
* CProgressBar::OnEraseBkgnd()
*
* Resize, progress control
*
*
* Arguments:
*
* none
*
* Return Value:
*
* status
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
BOOL CProgressBar::OnEraseBkgnd(CDC* pDC)
{
Resize();
return CProgressCtrl::OnEraseBkgnd(pDC);
}
/**************************************************************************\
* CProgressBar::SetPaneText()
*
* Set the text for the Pane (Status text for progress control)
*
*
* Arguments:
*
* nPane - pane number
* strText - Text to add to the status pane
*
* Return Value:
*
* status
*
* History:
*
* 2/14/1999 Original Version
*
\**************************************************************************/
BOOL CProgressBar::SetPaneText(int nPane, LPCTSTR strText)
{
CStatusBar* pStatusBar = GetStatusBar();
if(pStatusBar != NULL)
{
pStatusBar->SetPaneText(nPane,strText);
pStatusBar->UpdateWindow();
}
return TRUE;
}