Solved

customizing CStatusBar

Posted on 1998-09-21
8
621 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
Is Your Active Directory as Secure as You Think?

More than 75% of all records are compromised because of the loss or theft of a privileged credential. Experts have been exploring Active Directory infrastructure to identify key threats and establish best practices for keeping data safe. Attend this month’s webinar to learn more.

 
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

Is Your Active Directory as Secure as You Think?

More than 75% of all records are compromised because of the loss or theft of a privileged credential. Experts have been exploring Active Directory infrastructure to identify key threats and establish best practices for keeping data safe. Attend this month’s webinar to learn more.

Question has a verified solution.

If you are experiencing a similar issue, please ask a related question

Suggested Solutions

Title # Comments Views Activity
Host to IP 7 76
difference between String.subString() and String.subSequence() 6 169
changeXy challenge 13 67
wordcount challenge 11 106
Introduction: The undo support, implementing a stack. Continuing from the eigth article about sudoku.   We need a mechanism to keep track of the digits entered so as to implement an undo mechanism.  This should be a ‘Last In First Out’ collec…
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.
This is used to tweak the memory usage for your computer, it is used for servers more so than workstations but just be careful editing registry settings as it may cause irreversible results. I hold no responsibility for anything you do to the regist…

911 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

23 Experts available now in Live!

Get 1:1 Help Now