1 /////////////////////////////////////////////////////////////////////////////
2 // Program: wxWidgets Widgets Sample
3 // Name: samples/widgets/widgets.cpp
4 // Purpose: Sample showing most of the simple wxWidgets widgets
5 // Author: Vadim Zeitlin
8 // Copyright: (c) 2001 Vadim Zeitlin
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // for compilers that support precompilation, includes "wx/wx.h".
21 #include "wx/wxprec.h"
27 // for all others, include the necessary headers
35 #include "wx/button.h"
36 #include "wx/checkbox.h"
37 #include "wx/listbox.h"
38 #include "wx/statbox.h"
39 #include "wx/stattext.h"
40 #include "wx/textctrl.h"
41 #include "wx/msgdlg.h"
44 #include "wx/sysopt.h"
45 #include "wx/bookctrl.h"
46 #include "wx/treebook.h"
48 #include "wx/colordlg.h"
49 #include "wx/fontdlg.h"
50 #include "wx/textdlg.h"
51 #include "wx/imaglist.h"
52 #include "wx/wupdlock.h"
53 #include "wx/textcompleter.h"
55 #include "wx/persist/toplevel.h"
56 #include "wx/persist/treebook.h"
60 #include "../sample.xpm"
62 // ----------------------------------------------------------------------------
64 // ----------------------------------------------------------------------------
69 Widgets_ClearLog
= 100,
76 #endif // wxUSE_TOOLTIPS
89 Widgets_BorderDefault
,
91 Widgets_GlobalBusyCursor
,
95 Widgets_GoToPageLast
= Widgets_GoToPage
+ 100,
99 TextEntry_DisableAutoComplete
= TextEntry_Begin
,
100 TextEntry_AutoCompleteFixed
,
101 TextEntry_AutoCompleteFilenames
,
102 TextEntry_AutoCompleteDirectories
,
103 TextEntry_AutoCompleteCustom
,
109 const wxChar
*WidgetsCategories
[MAX_PAGES
] = {
110 #if defined(__WXUNIVERSAL__)
124 // ----------------------------------------------------------------------------
126 // ----------------------------------------------------------------------------
128 // Define a new application type, each program should derive a class from wxApp
129 class WidgetsApp
: public wxApp
132 // override base class virtuals
133 // ----------------------------
135 // this one is called on application startup and is a good place for the app
136 // initialization (doing it here and not in the ctor allows to have an error
137 // return: if OnInit() returns false, the application terminates)
138 virtual bool OnInit();
141 // Define a new frame type: this is going to be our main frame
142 class WidgetsFrame
: public wxFrame
146 WidgetsFrame(const wxString
& title
);
147 virtual ~WidgetsFrame();
152 void OnButtonClearLog(wxCommandEvent
& event
);
154 void OnExit(wxCommandEvent
& event
);
157 void OnPageChanging(WidgetsBookCtrlEvent
& event
);
158 void OnPageChanged(WidgetsBookCtrlEvent
& event
);
159 void OnGoToPage(wxCommandEvent
& event
);
162 void OnSetTooltip(wxCommandEvent
& event
);
163 #endif // wxUSE_TOOLTIPS
164 void OnSetFgCol(wxCommandEvent
& event
);
165 void OnSetBgCol(wxCommandEvent
& event
);
166 void OnSetPageBg(wxCommandEvent
& event
);
167 void OnSetFont(wxCommandEvent
& event
);
168 void OnEnable(wxCommandEvent
& event
);
169 void OnSetBorder(wxCommandEvent
& event
);
171 void OnToggleGlobalBusyCursor(wxCommandEvent
& event
);
172 void OnToggleBusyCursor(wxCommandEvent
& event
);
174 // wxTextEntry-specific tests
175 void OnDisableAutoComplete(wxCommandEvent
& event
);
176 void OnAutoCompleteFixed(wxCommandEvent
& event
);
177 void OnAutoCompleteFilenames(wxCommandEvent
& event
);
178 void OnAutoCompleteDirectories(wxCommandEvent
& event
);
179 void OnAutoCompleteCustom(wxCommandEvent
& event
);
181 void OnSetHint(wxCommandEvent
& event
);
183 void OnUpdateTextUI(wxUpdateUIEvent
& event
)
185 event
.Enable( CurrentPage()->GetTextEntry() != NULL
);
187 #endif // wxUSE_MENUS
189 // initialize the book: add all pages to it
192 // return the currently selected page (never NULL)
193 WidgetsPage
*CurrentPage();
196 // the panel containing everything
200 // the listbox for logging messages
201 wxListBox
*m_lboxLog
;
203 // the log target we use to redirect messages to the listbox
207 // the book containing the test pages
208 WidgetsBookCtrl
*m_book
;
211 // last chosen fg/bg colours and font
215 #endif // wxUSE_MENUS
217 // any class wishing to process wxWidgets events must use this macro
218 DECLARE_EVENT_TABLE()
222 // A log target which just redirects the messages to a listbox
223 class LboxLogger
: public wxLog
226 LboxLogger(wxListBox
*lbox
, wxLog
*logOld
)
229 //m_lbox->Disable(); -- looks ugly under MSW
233 virtual ~LboxLogger()
235 wxLog::SetActiveTarget(m_logOld
);
239 // implement sink functions
240 virtual void DoLogTextAtLevel(wxLogLevel level
, const wxString
& msg
)
242 if ( level
== wxLOG_Trace
)
245 m_logOld
->LogTextAtLevel(level
, msg
);
249 #ifdef __WXUNIVERSAL__
250 m_lbox
->AppendAndEnsureVisible(msg
);
251 #else // other ports don't have this method yet
253 m_lbox
->SetFirstItem(m_lbox
->GetCount() - 1);
257 // the control we use
260 // the old log target
266 WX_DEFINE_ARRAY_PTR(WidgetsPage
*, ArrayWidgetsPage
);
268 // ----------------------------------------------------------------------------
270 // ----------------------------------------------------------------------------
272 IMPLEMENT_APP(WidgetsApp
)
274 // ----------------------------------------------------------------------------
276 // ----------------------------------------------------------------------------
278 BEGIN_EVENT_TABLE(WidgetsFrame
, wxFrame
)
280 EVT_BUTTON(Widgets_ClearLog
, WidgetsFrame::OnButtonClearLog
)
282 EVT_BUTTON(Widgets_Quit
, WidgetsFrame::OnExit
)
285 EVT_MENU(Widgets_SetTooltip
, WidgetsFrame::OnSetTooltip
)
286 #endif // wxUSE_TOOLTIPS
289 EVT_WIDGETS_PAGE_CHANGING(wxID_ANY
, WidgetsFrame::OnPageChanging
)
290 EVT_MENU_RANGE(Widgets_GoToPage
, Widgets_GoToPageLast
,
291 WidgetsFrame::OnGoToPage
)
293 EVT_MENU(Widgets_SetFgColour
, WidgetsFrame::OnSetFgCol
)
294 EVT_MENU(Widgets_SetBgColour
, WidgetsFrame::OnSetBgCol
)
295 EVT_MENU(Widgets_SetPageBg
, WidgetsFrame::OnSetPageBg
)
296 EVT_MENU(Widgets_SetFont
, WidgetsFrame::OnSetFont
)
297 EVT_MENU(Widgets_Enable
, WidgetsFrame::OnEnable
)
299 EVT_MENU_RANGE(Widgets_BorderNone
, Widgets_BorderDefault
,
300 WidgetsFrame::OnSetBorder
)
302 EVT_MENU(Widgets_GlobalBusyCursor
, WidgetsFrame::OnToggleGlobalBusyCursor
)
303 EVT_MENU(Widgets_BusyCursor
, WidgetsFrame::OnToggleBusyCursor
)
305 EVT_MENU(TextEntry_DisableAutoComplete
, WidgetsFrame::OnDisableAutoComplete
)
306 EVT_MENU(TextEntry_AutoCompleteFixed
, WidgetsFrame::OnAutoCompleteFixed
)
307 EVT_MENU(TextEntry_AutoCompleteFilenames
, WidgetsFrame::OnAutoCompleteFilenames
)
308 EVT_MENU(TextEntry_AutoCompleteDirectories
, WidgetsFrame::OnAutoCompleteDirectories
)
309 EVT_MENU(TextEntry_AutoCompleteCustom
, WidgetsFrame::OnAutoCompleteCustom
)
311 EVT_MENU(TextEntry_SetHint
, WidgetsFrame::OnSetHint
)
313 EVT_UPDATE_UI_RANGE(TextEntry_Begin
, TextEntry_End
- 1,
314 WidgetsFrame::OnUpdateTextUI
)
316 EVT_MENU(wxID_EXIT
, WidgetsFrame::OnExit
)
317 #endif // wxUSE_MENUS
320 // ============================================================================
322 // ============================================================================
324 // ----------------------------------------------------------------------------
326 // ----------------------------------------------------------------------------
328 bool WidgetsApp::OnInit()
330 if ( !wxApp::OnInit() )
333 SetVendorName("wxWidgets_Samples");
335 // the reason for having these ifdef's is that I often run two copies of
336 // this sample side by side and it is useful to see which one is which
338 #if defined(__WXUNIVERSAL__)
339 title
= wxT("wxUniv/");
342 #if defined(__WXMSW__)
343 title
+= wxT("wxMSW");
344 #elif defined(__WXGTK__)
345 title
+= wxT("wxGTK");
346 #elif defined(__WXMAC__)
347 title
+= wxT("wxMAC");
348 #elif defined(__WXMOTIF__)
349 title
+= wxT("wxMOTIF");
351 title
+= wxT("wxWidgets");
354 wxFrame
*frame
= new WidgetsFrame(title
+ wxT(" widgets demo"));
360 // ----------------------------------------------------------------------------
361 // WidgetsFrame construction
362 // ----------------------------------------------------------------------------
364 WidgetsFrame::WidgetsFrame(const wxString
& title
)
365 : wxFrame(NULL
, wxID_ANY
, title
)
367 const bool sizeSet
= wxPersistentRegisterAndRestore(this, "Main");
369 // set the frame icon
370 SetIcon(wxICON(sample
));
380 // create the menubar
381 wxMenuBar
*mbar
= new wxMenuBar
;
382 wxMenu
*menuWidget
= new wxMenu
;
384 menuWidget
->Append(Widgets_SetTooltip
, wxT("Set &tooltip...\tCtrl-T"));
385 menuWidget
->AppendSeparator();
386 #endif // wxUSE_TOOLTIPS
387 menuWidget
->Append(Widgets_SetFgColour
, wxT("Set &foreground...\tCtrl-F"));
388 menuWidget
->Append(Widgets_SetBgColour
, wxT("Set &background...\tCtrl-B"));
389 menuWidget
->Append(Widgets_SetPageBg
, wxT("Set &page background...\tShift-Ctrl-B"));
390 menuWidget
->Append(Widgets_SetFont
, wxT("Set f&ont...\tCtrl-O"));
391 menuWidget
->AppendCheckItem(Widgets_Enable
, wxT("&Enable/disable\tCtrl-E"));
393 wxMenu
*menuBorders
= new wxMenu
;
394 menuBorders
->AppendRadioItem(Widgets_BorderDefault
, wxT("De&fault\tCtrl-Shift-9"));
395 menuBorders
->AppendRadioItem(Widgets_BorderNone
, wxT("&None\tCtrl-Shift-0"));
396 menuBorders
->AppendRadioItem(Widgets_BorderSimple
, wxT("&Simple\tCtrl-Shift-1"));
397 menuBorders
->AppendRadioItem(Widgets_BorderDouble
, wxT("&Double\tCtrl-Shift-2"));
398 menuBorders
->AppendRadioItem(Widgets_BorderStatic
, wxT("Stati&c\tCtrl-Shift-3"));
399 menuBorders
->AppendRadioItem(Widgets_BorderRaised
, wxT("&Raised\tCtrl-Shift-4"));
400 menuBorders
->AppendRadioItem(Widgets_BorderSunken
, wxT("S&unken\tCtrl-Shift-5"));
401 menuWidget
->AppendSubMenu(menuBorders
, wxT("Set &border"));
403 menuWidget
->AppendSeparator();
404 menuWidget
->AppendCheckItem(Widgets_GlobalBusyCursor
,
405 wxT("Toggle &global busy cursor\tCtrl-Shift-U"));
406 menuWidget
->AppendCheckItem(Widgets_BusyCursor
,
407 wxT("Toggle b&usy cursor\tCtrl-U"));
409 menuWidget
->AppendSeparator();
410 menuWidget
->Append(wxID_EXIT
, wxT("&Quit\tCtrl-Q"));
411 mbar
->Append(menuWidget
, wxT("&Widget"));
413 wxMenu
*menuTextEntry
= new wxMenu
;
414 menuTextEntry
->AppendRadioItem(TextEntry_DisableAutoComplete
,
415 wxT("&Disable auto-completion"));
416 menuTextEntry
->AppendRadioItem(TextEntry_AutoCompleteFixed
,
417 wxT("Fixed-&list auto-completion"));
418 menuTextEntry
->AppendRadioItem(TextEntry_AutoCompleteFilenames
,
419 wxT("&Files names auto-completion"));
420 menuTextEntry
->AppendRadioItem(TextEntry_AutoCompleteDirectories
,
421 wxT("&Directories names auto-completion"));
422 menuTextEntry
->AppendRadioItem(TextEntry_AutoCompleteCustom
,
423 wxT("&Custom auto-completion"));
424 menuTextEntry
->AppendSeparator();
425 menuTextEntry
->Append(TextEntry_SetHint
, "Set help &hint");
427 mbar
->Append(menuTextEntry
, wxT("&Text"));
431 mbar
->Check(Widgets_Enable
, true);
432 #endif // wxUSE_MENUS
435 m_panel
= new wxPanel(this, wxID_ANY
);
437 wxSizer
*sizerTop
= new wxBoxSizer(wxVERTICAL
);
439 // we have 2 panes: book with pages demonstrating the controls in the
440 // upper one and the log window with some buttons in the lower
442 int style
= wxBK_DEFAULT
;
443 // Uncomment to suppress page theme (draw in solid colour)
444 //style |= wxNB_NOPAGETHEME;
446 m_book
= new WidgetsBookCtrl(m_panel
, Widgets_BookCtrl
,
447 wxDefaultPosition
, wxDefaultSize
,
452 #ifndef __WXHANDHELD__
453 // the lower one only has the log listbox and a button to clear it
455 wxSizer
*sizerDown
= new wxStaticBoxSizer(
456 new wxStaticBox( m_panel
, wxID_ANY
, wxT("&Log window") ),
459 m_lboxLog
= new wxListBox(m_panel
, wxID_ANY
);
460 sizerDown
->Add(m_lboxLog
, 1, wxGROW
| wxALL
, 5);
461 sizerDown
->SetMinSize(100, 150);
463 wxSizer
*sizerDown
= new wxBoxSizer(wxVERTICAL
);
466 wxBoxSizer
*sizerBtns
= new wxBoxSizer(wxHORIZONTAL
);
469 btn
= new wxButton(m_panel
, Widgets_ClearLog
, wxT("Clear &log"));
471 sizerBtns
->Add(10, 0); // spacer
473 btn
= new wxButton(m_panel
, Widgets_Quit
, wxT("E&xit"));
475 sizerDown
->Add(sizerBtns
, 0, wxALL
| wxALIGN_RIGHT
, 5);
477 // put everything together
478 sizerTop
->Add(m_book
, 1, wxGROW
| (wxALL
& ~(wxTOP
| wxBOTTOM
)), 10);
479 sizerTop
->Add(0, 5, 0, wxGROW
); // spacer in between
480 sizerTop
->Add(sizerDown
, 0, wxGROW
| (wxALL
& ~wxTOP
), 10);
482 #else // !__WXHANDHELD__/__WXHANDHELD__
484 sizerTop
->Add(m_book
, 1, wxGROW
| wxALL
);
486 #endif // __WXHANDHELD__
488 m_panel
->SetSizer(sizerTop
);
490 const wxSize sizeMin
= m_panel
->GetBestSize();
492 SetClientSize(sizeMin
);
493 SetMinClientSize(sizeMin
);
495 #if USE_LOG && !defined(__WXCOCOA__)
496 // wxCocoa's listbox is too flakey to use for logging right now
497 // now that everything is created we can redirect the log messages to the
499 m_logTarget
= new LboxLogger(m_lboxLog
, wxLog::GetActiveTarget());
500 wxLog::SetActiveTarget(m_logTarget
);
504 void WidgetsFrame::InitBook()
506 #if USE_ICONS_IN_BOOK
507 wxImageList
*imageList
= new wxImageList(ICON_SIZE
, ICON_SIZE
);
509 wxImage
img(sample_xpm
);
510 imageList
->Add(wxBitmap(img
.Scale(ICON_SIZE
, ICON_SIZE
)));
512 wxImageList
*imageList
= NULL
;
516 WidgetsBookCtrl
*books
[MAX_PAGES
];
519 ArrayWidgetsPage pages
[MAX_PAGES
];
520 wxArrayString labels
[MAX_PAGES
];
522 wxMenu
*menuPages
= new wxMenu
;
523 unsigned int nPage
= 0, nFKey
= 0;
524 int cat
, imageId
= 1;
526 // we need to first create all pages and only then add them to the book
527 // as we need the image list first
529 // we also construct the pages menu during this first iteration
530 for ( cat
= 0; cat
< MAX_PAGES
; cat
++ )
533 nPage
++; // increase for parent page
535 books
[cat
] = new WidgetsBookCtrl(m_book
,
542 for ( WidgetsPageInfo
*info
= WidgetsPage::ms_widgetPages
;
544 info
= info
->GetNext() )
546 if( (info
->GetCategories() & ( 1 << cat
)) == 0)
549 WidgetsPage
*page
= (*info
->GetCtor())(
556 pages
[cat
].Add(page
);
558 labels
[cat
].Add(info
->GetLabel());
559 if ( cat
== ALL_PAGE
)
561 wxString
radioLabel(info
->GetLabel());
565 radioLabel
<< wxT("\tF" ) << nFKey
;
568 menuPages
->AppendRadioItem(
569 Widgets_GoToPage
+ nPage
,
573 // consider only for book in book architecture
579 // consider only for treebook architecture (with subpages)
585 GetMenuBar()->Append(menuPages
, wxT("&Page"));
587 #if USE_ICONS_IN_BOOK
588 m_book
->AssignImageList(imageList
);
591 for ( cat
= 0; cat
< MAX_PAGES
; cat
++ )
594 m_book
->AddPage(NULL
,WidgetsCategories
[cat
],false,0);
596 m_book
->AddPage(books
[cat
],WidgetsCategories
[cat
],false,0);
597 #if USE_ICONS_IN_BOOK
598 books
[cat
]->SetImageList(imageList
);
603 size_t count
= pages
[cat
].GetCount();
604 for ( size_t n
= 0; n
< count
; n
++ )
614 false, // don't select
621 wxEVT_COMMAND_WIDGETS_PAGE_CHANGED
,
622 wxWidgetsbookEventHandler(WidgetsFrame::OnPageChanged
) );
624 const bool pageSet
= wxPersistentRegisterAndRestore(m_book
);
627 // for treebook page #0 is empty parent page only so select the first page
628 // with some contents
630 m_book
->SetSelection(1);
632 // but ensure that the top of the tree is shown nevertheless
633 wxTreeCtrl
* const tree
= m_book
->GetTreeCtrl();
635 wxTreeItemIdValue cookie
;
636 tree
->EnsureVisible(tree
->GetFirstChild(tree
->GetRootItem(), cookie
));
640 // for other books set selection twice to force connected event handler
641 // to force lazy creation of initial visible content
642 m_book
->SetSelection(1);
643 m_book
->SetSelection(0);
645 #endif // USE_TREEBOOK
648 WidgetsPage
*WidgetsFrame::CurrentPage()
650 wxWindow
*page
= m_book
->GetCurrentPage();
653 WidgetsBookCtrl
*subBook
= wxStaticCast(page
, WidgetsBookCtrl
);
654 wxCHECK_MSG( subBook
, NULL
, wxT("no WidgetsBookCtrl?") );
656 page
= subBook
->GetCurrentPage();
657 #endif // !USE_TREEBOOK
659 return wxStaticCast(page
, WidgetsPage
);
662 WidgetsFrame::~WidgetsFrame()
669 // ----------------------------------------------------------------------------
670 // WidgetsFrame event handlers
671 // ----------------------------------------------------------------------------
673 void WidgetsFrame::OnExit(wxCommandEvent
& WXUNUSED(event
))
679 void WidgetsFrame::OnButtonClearLog(wxCommandEvent
& WXUNUSED(event
))
687 void WidgetsFrame::OnPageChanging(WidgetsBookCtrlEvent
& event
)
690 // don't allow selection of entries without pages (categories)
691 if ( !m_book
->GetPage(event
.GetSelection()) )
698 void WidgetsFrame::OnPageChanged(WidgetsBookCtrlEvent
& event
)
700 const int sel
= event
.GetSelection();
702 // adjust "Page" menu selection
703 wxMenuItem
*item
= GetMenuBar()->FindItem(Widgets_GoToPage
+ sel
);
707 GetMenuBar()->Check(Widgets_BusyCursor
, false);
709 // create the pages on demand, otherwise the sample startup is too slow as
710 // it creates hundreds of controls
711 WidgetsPage
*page
= CurrentPage();
712 if ( page
->GetChildren().empty() )
714 wxWindowUpdateLocker
noUpdates(page
);
715 page
->CreateContent();
717 page
->GetSizer()->Fit(page
);
719 WidgetsBookCtrl
*book
= wxStaticCast(page
->GetParent(), WidgetsBookCtrl
);
721 for ( size_t i
= 0; i
< book
->GetPageCount(); ++i
)
723 wxWindow
*page
= book
->GetPage(i
);
726 size
.IncTo(page
->GetSize());
735 void WidgetsFrame::OnGoToPage(wxCommandEvent
& event
)
738 m_book
->SetSelection(event
.GetId() - Widgets_GoToPage
);
740 m_book
->SetSelection(m_book
->GetPageCount()-1);
741 WidgetsBookCtrl
*book
= wxStaticCast(m_book
->GetCurrentPage(), WidgetsBookCtrl
);
742 book
->SetSelection(event
.GetId() - Widgets_GoToPage
);
748 void WidgetsFrame::OnSetTooltip(wxCommandEvent
& WXUNUSED(event
))
750 static wxString s_tip
= wxT("This is a tooltip");
752 wxTextEntryDialog dialog
755 wxT("Tooltip text (may use \\n, leave empty to remove): "),
756 wxT("Widgets sample"),
760 if ( dialog
.ShowModal() != wxID_OK
)
763 s_tip
= dialog
.GetValue();
764 s_tip
.Replace(wxT("\\n"), wxT("\n"));
766 WidgetsPage
*page
= CurrentPage();
768 const Widgets widgets
= page
->GetWidgets();
769 for ( Widgets::const_iterator it
= widgets
.begin();
773 (*it
)->SetToolTip(s_tip
);
777 #endif // wxUSE_TOOLTIPS
782 // Trivial wrapper for wxGetColourFromUser() which also does something even if
783 // the colour dialog is not available in the current build (which may happen
784 // for the ports in development and it is still useful to see how colours work)
785 wxColour
GetColourFromUser(wxWindow
*parent
, const wxColour
& colDefault
)
788 return wxGetColourFromUser(parent
, colDefault
);
789 #else // !wxUSE_COLOURDLG
790 if ( colDefault
== *wxBLACK
)
794 #endif // wxUSE_COLOURDLG/!wxUSE_COLOURDLG
797 } // anonymous namespace
799 void WidgetsFrame::OnSetFgCol(wxCommandEvent
& WXUNUSED(event
))
801 // allow for debugging the default colour the first time this is called
802 WidgetsPage
*page
= CurrentPage();
805 m_colFg
= page
->GetForegroundColour();
807 wxColour col
= GetColourFromUser(this, m_colFg
);
813 const Widgets widgets
= page
->GetWidgets();
814 for ( Widgets::const_iterator it
= widgets
.begin();
818 (*it
)->SetForegroundColour(m_colFg
);
823 void WidgetsFrame::OnSetBgCol(wxCommandEvent
& WXUNUSED(event
))
825 WidgetsPage
*page
= CurrentPage();
827 if ( !m_colBg
.IsOk() )
828 m_colBg
= page
->GetBackgroundColour();
830 wxColour col
= GetColourFromUser(this, m_colBg
);
836 const Widgets widgets
= page
->GetWidgets();
837 for ( Widgets::const_iterator it
= widgets
.begin();
841 (*it
)->SetBackgroundColour(m_colBg
);
846 void WidgetsFrame::OnSetPageBg(wxCommandEvent
& WXUNUSED(event
))
848 wxColour col
= GetColourFromUser(this, GetBackgroundColour());
852 CurrentPage()->SetBackgroundColour(col
);
853 CurrentPage()->Refresh();
856 void WidgetsFrame::OnSetFont(wxCommandEvent
& WXUNUSED(event
))
859 WidgetsPage
*page
= CurrentPage();
862 m_font
= page
->GetFont();
864 wxFont font
= wxGetFontFromUser(this, m_font
);
870 const Widgets widgets
= page
->GetWidgets();
871 for ( Widgets::const_iterator it
= widgets
.begin();
875 (*it
)->SetFont(m_font
);
879 // The best size of the widget could have changed after changing its font,
880 // so re-layout to show it correctly.
883 wxLogMessage(wxT("Font selection dialog not available in current build."));
887 void WidgetsFrame::OnEnable(wxCommandEvent
& event
)
889 const Widgets widgets
= CurrentPage()->GetWidgets();
890 for ( Widgets::const_iterator it
= widgets
.begin();
894 (*it
)->Enable(event
.IsChecked());
898 void WidgetsFrame::OnSetBorder(wxCommandEvent
& event
)
901 switch ( event
.GetId() )
903 case Widgets_BorderNone
: border
= wxBORDER_NONE
; break;
904 case Widgets_BorderStatic
: border
= wxBORDER_STATIC
; break;
905 case Widgets_BorderSimple
: border
= wxBORDER_SIMPLE
; break;
906 case Widgets_BorderRaised
: border
= wxBORDER_RAISED
; break;
907 case Widgets_BorderSunken
: border
= wxBORDER_SUNKEN
; break;
908 case Widgets_BorderDouble
: border
= wxBORDER_DOUBLE
; break;
911 wxFAIL_MSG( wxT("unknown border style") );
914 case Widgets_BorderDefault
: border
= wxBORDER_DEFAULT
; break;
917 WidgetsPage::ms_defaultFlags
&= ~wxBORDER_MASK
;
918 WidgetsPage::ms_defaultFlags
|= border
;
920 WidgetsPage
*page
= CurrentPage();
922 page
->RecreateWidget();
925 void WidgetsFrame::OnToggleGlobalBusyCursor(wxCommandEvent
& event
)
927 if ( event
.IsChecked() )
933 void WidgetsFrame::OnToggleBusyCursor(wxCommandEvent
& event
)
935 wxCursor
cursor(*(event
.IsChecked() ? wxHOURGLASS_CURSOR
936 : wxSTANDARD_CURSOR
));
938 const Widgets widgets
= CurrentPage()->GetWidgets();
939 for ( Widgets::const_iterator it
= widgets
.begin();
943 (*it
)->SetCursor(cursor
);
947 void WidgetsFrame::OnDisableAutoComplete(wxCommandEvent
& WXUNUSED(event
))
949 wxTextEntryBase
*entry
= CurrentPage()->GetTextEntry();
950 wxCHECK_RET( entry
, "menu item should be disabled" );
952 if ( entry
->AutoComplete(wxArrayString()) )
954 wxLogMessage("Disabled auto completion.");
958 wxLogMessage("AutoComplete() failed.");
962 void WidgetsFrame::OnAutoCompleteFixed(wxCommandEvent
& WXUNUSED(event
))
964 wxTextEntryBase
*entry
= CurrentPage()->GetTextEntry();
965 wxCHECK_RET( entry
, "menu item should be disabled" );
967 wxArrayString completion_choices
;
969 // add a few strings so a completion occurs on any letter typed
970 for ( char idxc
= 'a'; idxc
< 'z'; ++idxc
)
971 completion_choices
.push_back(wxString::Format("%c%c", idxc
, idxc
));
973 completion_choices
.push_back("is this string for test?");
974 completion_choices
.push_back("this is a test string");
975 completion_choices
.push_back("this is another test string");
976 completion_choices
.push_back("this string is for test");
978 if ( entry
->AutoComplete(completion_choices
) )
980 wxLogMessage("Enabled auto completion of a set of fixed strings.");
984 wxLogMessage("AutoComplete() failed.");
988 void WidgetsFrame::OnAutoCompleteFilenames(wxCommandEvent
& WXUNUSED(event
))
990 wxTextEntryBase
*entry
= CurrentPage()->GetTextEntry();
991 wxCHECK_RET( entry
, "menu item should be disabled" );
993 if ( entry
->AutoCompleteFileNames() )
995 wxLogMessage("Enabled auto completion of file names.");
999 wxLogMessage("AutoCompleteFileNames() failed.");
1003 void WidgetsFrame::OnAutoCompleteDirectories(wxCommandEvent
& WXUNUSED(event
))
1005 wxTextEntryBase
*entry
= CurrentPage()->GetTextEntry();
1006 wxCHECK_RET( entry
, "menu item should be disabled" );
1008 if ( entry
->AutoCompleteDirectories() )
1010 wxLogMessage("Enabled auto completion of directories.");
1014 wxLogMessage("AutoCompleteDirectories() failed.");
1018 void WidgetsFrame::OnAutoCompleteCustom(wxCommandEvent
& WXUNUSED(event
))
1020 wxTextEntryBase
*entry
= CurrentPage()->GetTextEntry();
1021 wxCHECK_RET( entry
, "menu item should be disabled" );
1023 // This is a simple (and hence rather useless) example of a custom
1024 // completer class that completes the first word (only) initially and only
1025 // build the list of the possible second words once the first word is
1026 // known. This allows to avoid building the full 676000 item list of
1027 // possible strings all at once as the we have 1000 possibilities for the
1028 // first word (000..999) and 676 (aa..zz) for the second one.
1029 class CustomTextCompleter
: public wxTextCompleterSimple
1032 virtual void GetCompletions(const wxString
& prefix
, wxArrayString
& res
)
1034 // This is used for illustrative purposes only and shows how many
1035 // completions we return every time when we're called.
1036 class LogCompletions
1039 LogCompletions(const wxString
& prefix
, const wxArrayString
& res
)
1047 wxLogMessage("Returning %lu possible completions for "
1049 m_res
.size(), m_prefix
);
1053 const wxString
& m_prefix
;
1054 const wxArrayString
& m_res
;
1055 } logCompletions(prefix
, res
);
1058 // Normally it doesn't make sense to complete empty control, there
1059 // are too many choices and listing them all wouldn't be helpful.
1060 if ( prefix
.empty() )
1063 // The only valid strings start with 3 digits so check for their
1064 // presence proposing to complete the remaining ones.
1065 if ( !wxIsdigit(prefix
[0]) )
1068 if ( prefix
.length() == 1 )
1070 for ( int i
= 0; i
< 10; i
++ )
1071 for ( int j
= 0; j
< 10; j
++ )
1072 res
.push_back(wxString::Format("%s%02d",
1076 else if ( !wxIsdigit(prefix
[1]) )
1079 if ( prefix
.length() == 2 )
1081 for ( int i
= 0; i
< 10; i
++ )
1082 res
.push_back(wxString::Format("%s%d", prefix
, i
));
1085 else if ( !wxIsdigit(prefix
[2]) )
1088 // Next we must have a space and two letters.
1089 wxString
prefix2(prefix
);
1090 if ( prefix
.length() == 3 )
1092 else if ( prefix
[3] != ' ' )
1095 if ( prefix2
.length() == 4 )
1097 for ( char c
= 'a'; c
<= 'z'; c
++ )
1098 for ( char d
= 'a'; d
<= 'z'; d
++ )
1099 res
.push_back(wxString::Format("%s%c%c", prefix2
, c
, d
));
1102 else if ( !wxIslower(prefix
[4]) )
1105 if ( prefix
.length() == 5 )
1107 for ( char c
= 'a'; c
<= 'z'; c
++ )
1108 res
.push_back(prefix
+ c
);
1113 if ( entry
->AutoComplete(new CustomTextCompleter
) )
1115 wxLogMessage("Enabled custom auto completer for \"NNN XX\" items "
1116 "(where N is a digit and X is a letter).");
1120 wxLogMessage("AutoComplete() failed.");
1124 void WidgetsFrame::OnSetHint(wxCommandEvent
& WXUNUSED(event
))
1126 wxTextEntryBase
*entry
= CurrentPage()->GetTextEntry();
1127 wxCHECK_RET( entry
, "menu item should be disabled" );
1129 static wxString
s_hint("Type here");
1131 hint
= wxGetTextFromUser("Text hint:", "Widgets sample", s_hint
, this);
1137 if ( entry
->SetHint(hint
) )
1139 wxLogMessage("Set hint to \"%s\".", hint
);
1143 wxLogMessage("Text hints not supported.");
1147 #endif // wxUSE_MENUS
1149 // ----------------------------------------------------------------------------
1151 // ----------------------------------------------------------------------------
1153 WidgetsPageInfo::WidgetsPageInfo(Constructor ctor
, const wxChar
*label
, int categories
)
1155 , m_categories(categories
)
1161 // dummy sorting: add and immediately sort in the list according to label
1162 if ( WidgetsPage::ms_widgetPages
)
1164 WidgetsPageInfo
*node_prev
= WidgetsPage::ms_widgetPages
;
1165 if ( wxStrcmp(label
, node_prev
->GetLabel().c_str()) < 0 )
1169 WidgetsPage::ms_widgetPages
= this;
1173 WidgetsPageInfo
*node_next
;
1176 node_next
= node_prev
->GetNext();
1179 // add if between two
1180 if ( wxStrcmp(label
, node_next
->GetLabel().c_str()) < 0 )
1182 node_prev
->SetNext(this);
1184 // force to break loop
1191 node_prev
->SetNext(this);
1194 node_prev
= node_next
;
1196 while ( node_next
);
1202 WidgetsPage::ms_widgetPages
= this;
1206 // ----------------------------------------------------------------------------
1208 // ----------------------------------------------------------------------------
1210 int WidgetsPage::ms_defaultFlags
= wxBORDER_DEFAULT
;
1211 WidgetsPageInfo
*WidgetsPage::ms_widgetPages
= NULL
;
1213 WidgetsPage::WidgetsPage(WidgetsBookCtrl
*book
,
1214 wxImageList
*imaglist
,
1215 const char *const icon
[])
1216 : wxPanel(book
, wxID_ANY
,
1217 wxDefaultPosition
, wxDefaultSize
,
1218 wxNO_FULL_REPAINT_ON_RESIZE
|
1222 #if USE_ICONS_IN_BOOK
1223 imaglist
->Add(wxBitmap(wxImage(icon
).Scale(ICON_SIZE
, ICON_SIZE
)));
1225 wxUnusedVar(imaglist
);
1230 wxSizer
*WidgetsPage::CreateSizerWithText(wxControl
*control
,
1232 wxTextCtrl
**ppText
)
1234 wxSizer
*sizerRow
= new wxBoxSizer(wxHORIZONTAL
);
1235 wxTextCtrl
*text
= new wxTextCtrl(this, id
, wxEmptyString
,
1236 wxDefaultPosition
, wxDefaultSize
, wxTE_PROCESS_ENTER
);
1238 sizerRow
->Add(control
, 0, wxRIGHT
| wxALIGN_CENTRE_VERTICAL
, 5);
1239 sizerRow
->Add(text
, 1, wxLEFT
| wxALIGN_CENTRE_VERTICAL
, 5);
1247 // create a sizer containing a label and a text ctrl
1248 wxSizer
*WidgetsPage::CreateSizerWithTextAndLabel(const wxString
& label
,
1250 wxTextCtrl
**ppText
)
1252 return CreateSizerWithText(new wxStaticText(this, wxID_ANY
, label
),
1256 // create a sizer containing a button and a text ctrl
1257 wxSizer
*WidgetsPage::CreateSizerWithTextAndButton(wxWindowID idBtn
,
1258 const wxString
& label
,
1260 wxTextCtrl
**ppText
)
1262 return CreateSizerWithText(new wxButton(this, idBtn
, label
), id
, ppText
);
1265 wxCheckBox
*WidgetsPage::CreateCheckBoxAndAddToSizer(wxSizer
*sizer
,
1266 const wxString
& label
,
1269 wxCheckBox
*checkbox
= new wxCheckBox(this, id
, label
);
1270 sizer
->Add(checkbox
, 0, wxLEFT
| wxRIGHT
, 5);
1271 sizer
->Add(0, 2, 0, wxGROW
); // spacer