Solved

customizing CStatusBar

Posted on 1998-09-21
8
599 Views
Last Modified: 2013-11-20
Is there an easy way to change the thickness of the borders
between panes in a status bar?  I found m_cxLeftBorder,
m_cxRightBorder, and m_cxDefaultGap as data in CControlBar.
I tried changing them, and the left and right border data
members work just as I expected...they alter the thickness
of the left and right borders.  I can't see that
m_cxDefaultGap changes anything, though.
0
Comment
Question by:eric_m
  • 4
  • 4
8 Comments
 
LVL 8

Expert Comment

by:Answers2000
ID: 1322378
There doesn't appear to be a way to do this (SB_GETBORDERS and MFC's GetBorders function let you retrieve the border size).

The way around.
1. Create a dummy item between each real item in the status bar, using SetParts (in CStatusBarCtrl - which you can get use GetStatusBarCtrl())
2. Set dummy parts to have no text and no border (SBT_NOBORDERS) using SetText member of CStatusBarCtrl.
3. Control width of "pseudo" borders by changing the width of the dummy items

OR

1. Override DrawItem for the control
2. Paint each item as you like with the borders you want
3. This is more work, but will give you finer control
0
 

Author Comment

by:eric_m
ID: 1322379
Could you please show me an example of how to override the
DrawItem to set the width?  Your answer doesn't provide me
with enough clues on how to do that.  Thanks!

0
 
LVL 8

Expert Comment

by:Answers2000
ID: 1322380
1. You need to set up the pane of the status bar as owner draw (for whichever pane you want).

GetStatusBarCtrl().SetText( (LPCTSTR)1 /* Pass as data to the owner draw proc */,  0 /* index of pane */, SBT_OWNERDRAW /* Flag for owner draw */ ) ;

A good place to do this is when you set up the status bar

2. Derive a status bar class from CStatusBar.  The Class Wizard can do this for you.  Create one of these rather than the default status bar.

3. In your derived class override DrawItem for the Status Bar (if you look in the help you will see only a DrawItem for CStatusBarCtrl, but the same function exists for CStatusBar).  In this function you get an LPDRAWITEMSTRUCT (pointer to a draw item structure) which contains the data you need to draw the pane including it's rectangle. [note:if class wizard probably won't let you do it, so you need to manually override].  It should look like this

in a .h

class CMyStatusBar : public CStatusBar
{
 // various stuff
public:
 virtual void DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct);
} ;

in .cpp

virtual void CMyStatusBar::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct)
{
 // <-- add you're code here
}

In the drawing code paint what ever you want.  The most important elements of lpDrawItemStruct are hDC (the DC to draw on) and rcItem (the rectangle of coordinates in pixels).

e.g.
virtual void CMyStatusBar::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct)
{
   // fill the pane with red for example
   CDC::FromHandle(lpDrawItemStruct->hDC)->FillSolidRect( &(lpDrawItemStruct->rcItem), RGB(255,0,0) ) ;
}

Inside the brackets you can add as much drawing code as you need using the standard drawing functions like Rectangle, LineTo, MoveToEx/MoveTo, DrawText etc.  

Hopefully this is enough to get you going.

0
 

Author Comment

by:eric_m
ID: 1322381
Answers2000:

Thanks for your help so far.  I'm still falling short,
though.

The default borders between panes in CStatusBar extend over
the shadows of the status pane.  They continue drawing one
pixel over and one pixel under the client area defined as
lpdrawitemstruct.rcItem.  I tried to do the same, but
Windows has probably set a clipping region so I don't draw
outside my little box.  In other words, I can't draw over
the bottom highlighted rectangle line, like the default
borders do.

The default borders, if they were thick enough, would line
up perfectly with splitter bars, which is what I'm trying to
do.  By drawing them myself, I'm left with a one pixel gap
in "no man's land" so to speak.

void MyStatusBar::DrawItem(LPDRAWITEMSTRUCT lpDrawItemStruct)
{
  CDC* pDC = CDC::FromHandle(lpDrawItemStruct->hDC);

  // arbitrarily chosen
  const int leftSide = 101;
  const int rightSide = 103;

  const int topSide = lpDrawItemStruct->rcItem.top - 1;
  const int bottomSide = lpDrawItemStruct->rcItem.bottom + 1;

  pDC->FillSolidRect(&(lpDrawItemStruct->rcItem), afxData.clrBtnFace);

  pDC->FillSolidRect(leftSide - 1,
                 topSide,
                 leftSide - 1,
                 bottomSide,
                 afxData.clrBtnHilite);


  pDC->FillSolidRect(leftSide,
                 topSide,
                 rightSide,
                 bottomSide,
                 afxData.clrBtnFace);
  pDC->FillSolidRect(rightSide + 1,
                 topSide,
                 rightSide + 1,
                 bottomSide,
                 afxData.clrBtnShadow);

  pDC->FillSolidRect(rightSide + 2,
                 lpDrawItemStruct->rcItem.top,
                 lpDrawItemStruct->rcItem.right,
                 lpDrawItemStruct->rcItem.bottom,
                 afxData.clrBtnFace);

}

0
How to run any project with ease

Manage projects of all sizes how you want. Great for personal to-do lists, project milestones, team priorities and launch plans.
- Combine task lists, docs, spreadsheets, and chat in one
- View and edit from mobile/offline
- Cut down on emails

 
LVL 8

Expert Comment

by:Answers2000
ID: 1322382
Hmm didn't know Windows would set a clipping region (it doesn't for most other owner draw controls)

I'm Not working with a sample, but could you SelectClipRgn at the start of DrawItem to remove the Clipping region, and then paint where-ever ?
0
 

Author Comment

by:eric_m
ID: 1322383
Answers2000:

Thanks for your help!  Adding to the region did the job.
Please re-answer so I can give you the points.  :)


  CRgn myRegion;
  myRegion.CreateRectRgn(leftSide-1,
                   topSide,
                   rightSide + 1,
                   bottomSide);

  pDC->SelectClipRgn(&myRegion, RGN_OR);

0
 
LVL 8

Accepted Solution

by:
Answers2000 earned 200 total points
ID: 1322384
Hurray solved at last!

And thanks for the points
0
 

Author Comment

by:eric_m
ID: 1322385
Thanks for your help and patience.  :)
0

Featured Post

6 Surprising Benefits of Threat Intelligence

All sorts of threat intelligence is available on the web. Intelligence you can learn from, and use to anticipate and prepare for future attacks.

Join & Write a Comment

Here is how to use MFC's automatic Radio Button handling in your dialog boxes and forms.  Beginner programmers usually start with a OnClick handler for each radio button and that's just not the right way to go.  MFC has a very cool system for handli…
Introduction: Dialogs (2) modeless dialog and a worker thread.  Handling data shared between threads.  Recursive functions. Continuing from the tenth article about sudoku.   Last article we worked with a modal dialog to help maintain informat…
This video will show you how to get GIT to work in Eclipse.   It will walk you through how to install the EGit plugin in eclipse and how to checkout an existing repository.
In this tutorial you'll learn about bandwidth monitoring with flows and packet sniffing with our network monitoring solution PRTG Network Monitor (https://www.paessler.com/prtg). If you're interested in additional methods for monitoring bandwidt…

746 members asked questions and received personalized solutions in the past 7 days.

Join the community of 500,000 technology professionals and ask your questions.

Join & Ask a Question

Need Help in Real-Time?

Connect with top rated Experts

12 Experts available now in Live!

Get 1:1 Help Now