1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: src/generic/infobar.cpp
3 // Purpose: generic wxInfoBar implementation
4 // Author: Vadim Zeitlin
6 // RCS-ID: $Id: wxhead.cpp,v 1.10 2009-06-29 10:23:04 zeitlin Exp $
7 // Copyright: (c) 2009 Vadim Zeitlin <vadim@wxwidgets.org>
8 // Licence: wxWindows licence
9 ///////////////////////////////////////////////////////////////////////////////
11 // ============================================================================
13 // ============================================================================
15 // ----------------------------------------------------------------------------
17 // ----------------------------------------------------------------------------
19 // for compilers that support precompilation, includes "wx.h".
20 #include "wx/wxprec.h"
29 #include "wx/bmpbuttn.h"
30 #include "wx/button.h"
31 #include "wx/settings.h"
32 #include "wx/statbmp.h"
33 #include "wx/stattext.h"
36 #include "wx/infobar.h"
38 #include "wx/artprov.h"
39 #include "wx/renderer.h"
40 #include "wx/scopeguard.h"
43 BEGIN_EVENT_TABLE(wxInfoBarGeneric
, wxInfoBarBase
)
44 EVT_BUTTON(wxID_ANY
, wxInfoBarGeneric::OnButton
)
47 // ----------------------------------------------------------------------------
49 // ----------------------------------------------------------------------------
54 #ifdef wxHAS_DRAW_TITLE_BAR_BITMAP
57 GetCloseButtonBitmap(wxWindow
*win
,
59 const wxColour
& colBg
,
64 dc
.SetBackground(colBg
);
66 wxRendererNative::Get().
67 DrawTitleBarBitmap(win
, dc
, size
, wxTITLEBAR_BUTTON_CLOSE
, flags
);
71 #endif // wxHAS_DRAW_TITLE_BAR_BITMAP
73 } // anonymous namespace
75 // ============================================================================
77 // ============================================================================
79 void wxInfoBarGeneric::Init()
85 m_showEffect
= wxSHOW_EFFECT_SLIDE_TO_BOTTOM
;
86 m_hideEffect
= wxSHOW_EFFECT_SLIDE_TO_TOP
;
88 // use default effect duration
92 bool wxInfoBarGeneric::Create(wxWindow
*parent
, wxWindowID winid
)
94 // calling Hide() before Create() ensures that we're created initially
97 if ( !wxWindow::Create(parent
, winid
) )
100 // use special, easy to notice, colours
101 const wxColour colBg
= wxSystemSettings::GetColour(wxSYS_COLOUR_INFOBK
);
102 SetBackgroundColour(colBg
);
103 SetOwnForegroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_INFOTEXT
));
105 // create the controls: icon, text and the button to dismiss the
108 // the icon is not shown unless it's assigned a valid bitmap
109 m_icon
= new wxStaticBitmap(this, wxID_ANY
, wxNullBitmap
);
111 m_text
= new wxStaticText(this, wxID_ANY
, "");
113 #ifdef wxHAS_DRAW_TITLE_BAR_BITMAP
114 const wxSize sizeBmp
= wxArtProvider::GetSizeHint(wxART_FRAME_ICON
);
115 wxBitmap bmp
= GetCloseButtonBitmap(this, sizeBmp
, colBg
);
116 #else // !wxHAS_DRAW_TITLE_BAR_BITMAP
117 wxBitmap bmp
= wxArtProvider::GetBitmap(wxART_CLOSE
, wxART_MENU
);
118 #endif // wxHAS_DRAW_TITLE_BAR_BITMAP
119 m_button
= new wxBitmapButton
129 #ifdef wxHAS_DRAW_TITLE_BAR_BITMAP
130 m_button
->SetBitmapPressed(
131 GetCloseButtonBitmap(this, sizeBmp
, colBg
, wxCONTROL_PRESSED
));
133 m_button
->SetBitmapCurrent(
134 GetCloseButtonBitmap(this, sizeBmp
, colBg
, wxCONTROL_CURRENT
));
135 #endif // wxHAS_DRAW_TITLE_BAR_BITMAP
137 m_button
->SetBackgroundColour(colBg
);
138 m_button
->SetToolTip(_("Hide this notification message."));
140 // center the text inside the sizer with an icon to the left of it and a
141 // button at the very right
143 // NB: AddButton() relies on the button being the last control in the sizer
144 // and being preceded by a spacer
145 wxSizer
* const sizer
= new wxBoxSizer(wxHORIZONTAL
);
146 sizer
->Add(m_icon
, wxSizerFlags().Centre().Border());
147 sizer
->Add(m_text
, wxSizerFlags().Centre());
148 sizer
->AddStretchSpacer();
149 sizer
->Add(m_button
, wxSizerFlags().Centre().Border());
155 bool wxInfoBarGeneric::SetFont(const wxFont
& font
)
157 if ( !wxInfoBarBase::SetFont(font
) )
160 // check that we're not called before Create()
162 m_text
->SetFont(font
);
167 void wxInfoBarGeneric::UpdateParent()
169 wxWindow
* const parent
= wxGetTopLevelParent(GetParent());
173 void wxInfoBarGeneric::ChangeParentBackground()
175 wxWindow
* const parent
= GetParent();
176 m_origParentBgCol
= parent
->GetBackgroundColour();
178 wxSizer
* const sizer
= GetContainingSizer();
182 wxWindow
*sibling
= NULL
;
183 for ( wxSizerItemList::compatibility_iterator
184 node
= sizer
->GetChildren().GetFirst();
186 node
= node
->GetNext() )
188 if ( node
->GetData()->GetWindow() == this )
190 // find the next window following us
191 for ( node
= node
->GetNext();
193 node
= node
->GetNext() )
195 wxSizerItem
* const item
= node
->GetData();
196 if ( item
->IsWindow() )
198 sibling
= item
->GetWindow();
208 parent
->SetOwnBackgroundColour(sibling
->GetBackgroundColour());
211 void wxInfoBarGeneric::RestoreParentBackground()
213 GetParent()->SetOwnBackgroundColour(m_origParentBgCol
);
216 void wxInfoBarGeneric::DoHide()
218 ChangeParentBackground();
219 wxON_BLOCK_EXIT_THIS0( wxInfoBarGeneric::RestoreParentBackground
);
221 HideWithEffect(m_hideEffect
, m_effectDuration
);
225 void wxInfoBarGeneric::DoShow()
227 // re-layout the parent first so that the window expands into an already
228 // unoccupied by the other controls area: for this we need to change our
229 // internal visibility flag to force Layout() to take us into account (an
230 // alternative solution to this hack would be to temporarily set
231 // wxRESERVE_SPACE_EVEN_IF_HIDDEN flag but it's not really batter)
233 // just change the internal flag indicating that the window is visible,
234 // without really showing it
235 wxWindowBase::Show();
237 // an extra hack: we want the temporarily uncovered area in which we're
238 // going to expand to look like part of this sibling for a better effect so
239 // temporarily change the background of our parent to the same colour
240 ChangeParentBackground();
241 wxON_BLOCK_EXIT_THIS0( wxInfoBarGeneric::RestoreParentBackground
);
243 // adjust the parent layout to account for us
246 // reset the flag back before really showing the window or it wouldn't be
247 // shown at all because it would believe itself already visible
248 wxWindowBase::Show(false);
251 // finally do really show the window.
252 ShowWithEffect(m_showEffect
, m_effectDuration
);
255 void wxInfoBarGeneric::ShowMessage(const wxString
& msg
, int flags
)
257 // first update the controls
258 const int icon
= flags
& wxICON_MASK
;
259 if ( !icon
|| (icon
== wxICON_NONE
) )
263 else // do show an icon
265 m_icon
->SetBitmap(wxArtProvider::GetBitmap(
266 wxArtProvider::GetMessageBoxIconId(flags
),
271 // notice the use of EscapeMnemonics() to ensure that "&" come through
273 m_text
->SetLabel(wxControl::EscapeMnemonics(msg
));
276 // then show this entire window if not done yet
281 else // we're already shown
283 // just update the layout to correspond to the new message
288 void wxInfoBarGeneric::Dismiss()
293 void wxInfoBarGeneric::AddButton(wxWindowID btnid
, const wxString
& label
)
295 wxSizer
* const sizer
= GetSizer();
296 wxCHECK_RET( sizer
, "must be created first" );
298 // user-added buttons replace the standard close button so remove it if we
299 // hadn't done it yet
300 if ( sizer
->Detach(m_button
) )
305 sizer
->Add(new wxButton(this, btnid
, label
),
306 wxSizerFlags().Centre().DoubleBorder());
309 void wxInfoBarGeneric::RemoveButton(wxWindowID btnid
)
311 wxSizer
* const sizer
= GetSizer();
312 wxCHECK_RET( sizer
, "must be created first" );
314 // iterate over the sizer items in reverse order to find the last added
315 // button with this id (ids of all buttons should be unique anyhow but if
316 // they are repeated removing the last added one probably makes more sense)
317 const wxSizerItemList
& items
= sizer
->GetChildren();
318 for ( wxSizerItemList::compatibility_iterator node
= items
.GetLast();
319 node
!= items
.GetFirst();
320 node
= node
->GetPrevious() )
322 const wxSizerItem
* const item
= node
->GetData();
324 // if we reached the spacer separating the buttons from the text
325 // preceding them without finding our button, it must mean it's not
327 if ( item
->IsSpacer() )
329 wxFAIL_MSG( wxString::Format("button with id %d not found", btnid
) );
333 // check if we found our button
334 if ( item
->GetWindow()->GetId() == btnid
)
336 delete item
->GetWindow();
341 // check if there are any custom buttons left
342 if ( sizer
->GetChildren().GetLast()->GetData()->IsSpacer() )
344 // if the last item is the spacer, none are left so restore the
345 // standard close button
346 sizer
->Add(m_button
, wxSizerFlags().Centre().DoubleBorder());
351 void wxInfoBarGeneric::OnButton(wxCommandEvent
& WXUNUSED(event
))
356 #endif // wxUSE_INFOBAR