]> git.saurik.com Git - wxWidgets.git/blob - samples/widgets/widgets.cpp
glibc's vswprintf doesn't nul terminate on truncation.
[wxWidgets.git] / samples / widgets / widgets.cpp
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
6 // Created: 27.03.01
7 // Id: $Id$
8 // Copyright: (c) 2001 Vadim Zeitlin
9 // License: wxWindows license
10 /////////////////////////////////////////////////////////////////////////////
11
12 // ============================================================================
13 // declarations
14 // ============================================================================
15
16 // ----------------------------------------------------------------------------
17 // headers
18 // ----------------------------------------------------------------------------
19
20 // for compilers that support precompilation, includes "wx/wx.h".
21 #include "wx/wxprec.h"
22
23 #ifdef __BORLANDC__
24 #pragma hdrstop
25 #endif
26
27 // for all others, include the necessary headers
28 #ifndef WX_PRECOMP
29 #include "wx/app.h"
30 #include "wx/log.h"
31 #include "wx/frame.h"
32 #include "wx/menu.h"
33
34 #include "wx/button.h"
35 #include "wx/checkbox.h"
36 #include "wx/listbox.h"
37 #include "wx/statbox.h"
38 #include "wx/stattext.h"
39 #include "wx/textctrl.h"
40 #include "wx/msgdlg.h"
41 #endif
42
43 #include "wx/sysopt.h"
44 #include "wx/bookctrl.h"
45 #include "wx/treebook.h"
46 #include "wx/sizer.h"
47 #include "wx/colordlg.h"
48 #include "wx/fontdlg.h"
49 #include "wx/textdlg.h"
50 #include "wx/imaglist.h"
51 #include "wx/wupdlock.h"
52
53 #include "widgets.h"
54
55 #include "../sample.xpm"
56
57 // ----------------------------------------------------------------------------
58 // constants
59 // ----------------------------------------------------------------------------
60
61 // control ids
62 enum
63 {
64 Widgets_ClearLog = 100,
65 Widgets_Quit,
66
67 Widgets_BookCtrl,
68
69 #if wxUSE_TOOLTIPS
70 Widgets_SetTooltip,
71 #endif // wxUSE_TOOLTIPS
72 Widgets_SetFgColour,
73 Widgets_SetBgColour,
74 Widgets_SetFont,
75 Widgets_Enable,
76
77 Widgets_BorderNone,
78 Widgets_BorderStatic,
79 Widgets_BorderSimple,
80 Widgets_BorderRaised,
81 Widgets_BorderSunken,
82 Widgets_BorderDouble,
83 Widgets_BorderDefault,
84
85 Widgets_GlobalBusyCursor,
86 Widgets_BusyCursor,
87
88 Widgets_GoToPage,
89 Widgets_GoToPageLast = Widgets_GoToPage + 100
90 };
91
92 const wxChar *WidgetsCategories[MAX_PAGES] = {
93 #if defined(__WXUNIVERSAL__)
94 wxT("Universal"),
95 #else
96 wxT("Native"),
97 #endif
98 wxT("Generic"),
99 wxT("Pickers"),
100 wxT("Comboboxes"),
101 wxT("With items"),
102 wxT("Editable"),
103 wxT("Books"),
104 wxT("All controls")
105 };
106
107 // ----------------------------------------------------------------------------
108 // our classes
109 // ----------------------------------------------------------------------------
110
111 // Define a new application type, each program should derive a class from wxApp
112 class WidgetsApp : public wxApp
113 {
114 public:
115 // override base class virtuals
116 // ----------------------------
117
118 // this one is called on application startup and is a good place for the app
119 // initialization (doing it here and not in the ctor allows to have an error
120 // return: if OnInit() returns false, the application terminates)
121 virtual bool OnInit();
122 };
123
124 // Define a new frame type: this is going to be our main frame
125 class WidgetsFrame : public wxFrame
126 {
127 public:
128 // ctor(s) and dtor
129 WidgetsFrame(const wxString& title);
130 virtual ~WidgetsFrame();
131
132 protected:
133 // event handlers
134 #if USE_LOG
135 void OnButtonClearLog(wxCommandEvent& event);
136 #endif // USE_LOG
137 void OnExit(wxCommandEvent& event);
138
139 #if wxUSE_MENUS
140 void OnPageChanging(WidgetsBookCtrlEvent& event);
141 void OnPageChanged(WidgetsBookCtrlEvent& event);
142 void OnGoToPage(wxCommandEvent& event);
143
144 #if wxUSE_TOOLTIPS
145 void OnSetTooltip(wxCommandEvent& event);
146 #endif // wxUSE_TOOLTIPS
147 void OnSetFgCol(wxCommandEvent& event);
148 void OnSetBgCol(wxCommandEvent& event);
149 void OnSetFont(wxCommandEvent& event);
150 void OnEnable(wxCommandEvent& event);
151 void OnSetBorder(wxCommandEvent& event);
152
153 void OnToggleGlobalBusyCursor(wxCommandEvent& event);
154 void OnToggleBusyCursor(wxCommandEvent& event);
155 #endif // wxUSE_MENUS
156
157 // initialize the book: add all pages to it
158 void InitBook();
159
160 // return the currently selected page (never NULL)
161 WidgetsPage *CurrentPage();
162
163 private:
164 // the panel containing everything
165 wxPanel *m_panel;
166
167 #if USE_LOG
168 // the listbox for logging messages
169 wxListBox *m_lboxLog;
170
171 // the log target we use to redirect messages to the listbox
172 wxLog *m_logTarget;
173 #endif // USE_LOG
174
175 // the book containing the test pages
176 WidgetsBookCtrl *m_book;
177
178 #if wxUSE_MENUS
179 // last chosen fg/bg colours and font
180 wxColour m_colFg,
181 m_colBg;
182 wxFont m_font;
183 #endif // wxUSE_MENUS
184
185 // any class wishing to process wxWidgets events must use this macro
186 DECLARE_EVENT_TABLE()
187 };
188
189 #if USE_LOG
190 // A log target which just redirects the messages to a listbox
191 class LboxLogger : public wxLog
192 {
193 public:
194 LboxLogger(wxListBox *lbox, wxLog *logOld)
195 {
196 m_lbox = lbox;
197 //m_lbox->Disable(); -- looks ugly under MSW
198 m_logOld = logOld;
199 }
200
201 virtual ~LboxLogger()
202 {
203 wxLog::SetActiveTarget(m_logOld);
204 }
205
206 private:
207 // implement sink functions
208 virtual void DoLog(wxLogLevel level, const wxChar *szString, time_t t)
209 {
210 // don't put trace messages into listbox or we can get into infinite
211 // recursion
212 if ( level == wxLOG_Trace )
213 {
214 if ( m_logOld )
215 {
216 // cast is needed to call protected method
217 ((LboxLogger *)m_logOld)->DoLog(level, szString, t);
218 }
219 }
220 else
221 {
222 wxLog::DoLog(level, szString, t);
223 }
224 }
225
226 virtual void DoLogString(const wxChar *szString, time_t WXUNUSED(t))
227 {
228 wxString msg;
229 TimeStamp(&msg);
230 msg += szString;
231
232 #ifdef __WXUNIVERSAL__
233 m_lbox->AppendAndEnsureVisible(msg);
234 #else // other ports don't have this method yet
235 m_lbox->Append(msg);
236 m_lbox->SetFirstItem(m_lbox->GetCount() - 1);
237 #endif
238 }
239
240 // the control we use
241 wxListBox *m_lbox;
242
243 // the old log target
244 wxLog *m_logOld;
245 };
246 #endif // USE_LOG
247
248 // array of pages
249 WX_DEFINE_ARRAY_PTR(WidgetsPage *, ArrayWidgetsPage);
250
251 // ----------------------------------------------------------------------------
252 // misc macros
253 // ----------------------------------------------------------------------------
254
255 IMPLEMENT_APP(WidgetsApp)
256
257 // ----------------------------------------------------------------------------
258 // event tables
259 // ----------------------------------------------------------------------------
260
261 BEGIN_EVENT_TABLE(WidgetsFrame, wxFrame)
262 #if USE_LOG
263 EVT_BUTTON(Widgets_ClearLog, WidgetsFrame::OnButtonClearLog)
264 #endif // USE_LOG
265 EVT_BUTTON(Widgets_Quit, WidgetsFrame::OnExit)
266
267 #if wxUSE_TOOLTIPS
268 EVT_MENU(Widgets_SetTooltip, WidgetsFrame::OnSetTooltip)
269 #endif // wxUSE_TOOLTIPS
270
271 #if wxUSE_MENUS
272 EVT_WIDGETS_PAGE_CHANGING(wxID_ANY, WidgetsFrame::OnPageChanging)
273 EVT_WIDGETS_PAGE_CHANGED(wxID_ANY, WidgetsFrame::OnPageChanged)
274 EVT_MENU_RANGE(Widgets_GoToPage, Widgets_GoToPageLast,
275 WidgetsFrame::OnGoToPage)
276
277 EVT_MENU(Widgets_SetFgColour, WidgetsFrame::OnSetFgCol)
278 EVT_MENU(Widgets_SetBgColour, WidgetsFrame::OnSetBgCol)
279 EVT_MENU(Widgets_SetFont, WidgetsFrame::OnSetFont)
280 EVT_MENU(Widgets_Enable, WidgetsFrame::OnEnable)
281
282 EVT_MENU_RANGE(Widgets_BorderNone, Widgets_BorderDefault,
283 WidgetsFrame::OnSetBorder)
284
285 EVT_MENU(Widgets_GlobalBusyCursor, WidgetsFrame::OnToggleGlobalBusyCursor)
286 EVT_MENU(Widgets_BusyCursor, WidgetsFrame::OnToggleBusyCursor)
287
288 EVT_MENU(wxID_EXIT, WidgetsFrame::OnExit)
289 #endif // wxUSE_MENUS
290 END_EVENT_TABLE()
291
292 // ============================================================================
293 // implementation
294 // ============================================================================
295
296 // ----------------------------------------------------------------------------
297 // app class
298 // ----------------------------------------------------------------------------
299
300 bool WidgetsApp::OnInit()
301 {
302 if ( !wxApp::OnInit() )
303 return false;
304
305 // the reason for having these ifdef's is that I often run two copies of
306 // this sample side by side and it is useful to see which one is which
307 wxString title;
308 #if defined(__WXUNIVERSAL__)
309 title = _T("wxUniv/");
310 #endif
311
312 #if defined(__WXMSW__)
313 title += _T("wxMSW");
314 #elif defined(__WXGTK__)
315 title += _T("wxGTK");
316 #elif defined(__WXMAC__)
317 title += _T("wxMAC");
318 #elif defined(__WXMOTIF__)
319 title += _T("wxMOTIF");
320 #else
321 title += _T("wxWidgets");
322 #endif
323
324 wxFrame *frame = new WidgetsFrame(title + _T(" widgets demo"));
325 frame->Show();
326
327 //wxLog::AddTraceMask(_T("listbox"));
328 //wxLog::AddTraceMask(_T("scrollbar"));
329 //wxLog::AddTraceMask(_T("focus"));
330
331 return true;
332 }
333
334 // ----------------------------------------------------------------------------
335 // WidgetsFrame construction
336 // ----------------------------------------------------------------------------
337
338 WidgetsFrame::WidgetsFrame(const wxString& title)
339 : wxFrame(NULL, wxID_ANY, title)
340 {
341 // set the frame icon
342 SetIcon(wxICON(sample));
343
344 // init everything
345 #if USE_LOG
346 m_lboxLog = (wxListBox *)NULL;
347 m_logTarget = (wxLog *)NULL;
348 #endif // USE_LOG
349 m_book = (WidgetsBookCtrl *)NULL;
350
351 #if wxUSE_MENUS
352 // create the menubar
353 wxMenuBar *mbar = new wxMenuBar;
354 wxMenu *menuWidget = new wxMenu;
355 #if wxUSE_TOOLTIPS
356 menuWidget->Append(Widgets_SetTooltip, _T("Set &tooltip...\tCtrl-T"));
357 menuWidget->AppendSeparator();
358 #endif // wxUSE_TOOLTIPS
359 menuWidget->Append(Widgets_SetFgColour, _T("Set &foreground...\tCtrl-F"));
360 menuWidget->Append(Widgets_SetBgColour, _T("Set &background...\tCtrl-B"));
361 menuWidget->Append(Widgets_SetFont, _T("Set f&ont...\tCtrl-O"));
362 menuWidget->AppendCheckItem(Widgets_Enable, _T("&Enable/disable\tCtrl-E"));
363
364 wxMenu *menuBorders = new wxMenu;
365 menuBorders->AppendRadioItem(Widgets_BorderDefault, _T("De&fault\tCtrl-Shift-9"));
366 menuBorders->AppendRadioItem(Widgets_BorderNone, _T("&None\tCtrl-Shift-0"));
367 menuBorders->AppendRadioItem(Widgets_BorderSimple, _T("&Simple\tCtrl-Shift-1"));
368 menuBorders->AppendRadioItem(Widgets_BorderDouble, _T("&Double\tCtrl-Shift-2"));
369 menuBorders->AppendRadioItem(Widgets_BorderStatic, _T("Stati&c\tCtrl-Shift-3"));
370 menuBorders->AppendRadioItem(Widgets_BorderRaised, _T("&Raised\tCtrl-Shift-4"));
371 menuBorders->AppendRadioItem(Widgets_BorderSunken, _T("S&unken\tCtrl-Shift-5"));
372 menuWidget->AppendSubMenu(menuBorders, _T("Set &border"));
373
374 menuWidget->AppendSeparator();
375 menuWidget->AppendCheckItem(Widgets_GlobalBusyCursor,
376 _T("Toggle &global busy cursor\tCtrl-Shift-U"));
377 menuWidget->AppendCheckItem(Widgets_BusyCursor,
378 _T("Toggle b&usy cursor\tCtrl-U"));
379
380 menuWidget->AppendSeparator();
381 menuWidget->Append(wxID_EXIT, _T("&Quit\tCtrl-Q"));
382 mbar->Append(menuWidget, _T("&Widget"));
383 SetMenuBar(mbar);
384
385 mbar->Check(Widgets_Enable, true);
386 #endif // wxUSE_MENUS
387
388 // create controls
389 m_panel = new wxPanel(this, wxID_ANY);
390
391 wxSizer *sizerTop = new wxBoxSizer(wxVERTICAL);
392
393 // we have 2 panes: book with pages demonstrating the controls in the
394 // upper one and the log window with some buttons in the lower
395
396 int style = wxBK_DEFAULT;
397 // Uncomment to suppress page theme (draw in solid colour)
398 //style |= wxNB_NOPAGETHEME;
399
400 m_book = new WidgetsBookCtrl(m_panel, Widgets_BookCtrl, wxDefaultPosition,
401 #ifdef __WXMOTIF__
402 wxSize(500, wxDefaultCoord), // under Motif, height is a function of the width...
403 #else
404 wxDefaultSize,
405 #endif
406 style);
407 InitBook();
408
409 #ifndef __WXHANDHELD__
410 // the lower one only has the log listbox and a button to clear it
411 #if USE_LOG
412 wxSizer *sizerDown = new wxStaticBoxSizer(
413 new wxStaticBox( m_panel, wxID_ANY, _T("&Log window") ),
414 wxVERTICAL);
415
416 m_lboxLog = new wxListBox(m_panel, wxID_ANY);
417 sizerDown->Add(m_lboxLog, 1, wxGROW | wxALL, 5);
418 sizerDown->SetMinSize(100, 150);
419 #else
420 wxSizer *sizerDown = new wxBoxSizer(wxVERTICAL);
421 #endif // USE_LOG
422
423 wxBoxSizer *sizerBtns = new wxBoxSizer(wxHORIZONTAL);
424 wxButton *btn;
425 #if USE_LOG
426 btn = new wxButton(m_panel, Widgets_ClearLog, _T("Clear &log"));
427 sizerBtns->Add(btn);
428 sizerBtns->Add(10, 0); // spacer
429 #endif // USE_LOG
430 btn = new wxButton(m_panel, Widgets_Quit, _T("E&xit"));
431 sizerBtns->Add(btn);
432 sizerDown->Add(sizerBtns, 0, wxALL | wxALIGN_RIGHT, 5);
433
434 // put everything together
435 sizerTop->Add(m_book, 1, wxGROW | (wxALL & ~(wxTOP | wxBOTTOM)), 10);
436 sizerTop->Add(0, 5, 0, wxGROW); // spacer in between
437 sizerTop->Add(sizerDown, 0, wxGROW | (wxALL & ~wxTOP), 10);
438
439 #else // !__WXHANDHELD__/__WXHANDHELD__
440
441 sizerTop->Add(m_book, 1, wxGROW | wxALL );
442
443 #endif // __WXHANDHELD__
444
445 m_panel->SetSizer(sizerTop);
446
447 sizerTop->Fit(this);
448 sizerTop->SetSizeHints(this);
449
450 #if USE_LOG && !defined(__WXCOCOA__)
451 // wxCocoa's listbox is too flakey to use for logging right now
452 // now that everything is created we can redirect the log messages to the
453 // listbox
454 m_logTarget = new LboxLogger(m_lboxLog, wxLog::GetActiveTarget());
455 wxLog::SetActiveTarget(m_logTarget);
456 #endif
457 }
458
459 void WidgetsFrame::InitBook()
460 {
461 #if USE_ICONS_IN_BOOK
462 wxImageList *imageList = new wxImageList(32, 32);
463
464 imageList->Add(wxBitmap(sample_xpm));
465 #else
466 wxImageList *imageList = NULL;
467 #endif
468
469 #if !USE_TREEBOOK
470 WidgetsBookCtrl *books[MAX_PAGES];
471 #endif
472
473 ArrayWidgetsPage pages[MAX_PAGES];
474 wxArrayString labels[MAX_PAGES];
475
476 wxMenu *menuPages = new wxMenu;
477 unsigned int nPage = 0, nFKey = 0;
478 int cat, imageId = 1;
479
480 // we need to first create all pages and only then add them to the book
481 // as we need the image list first
482 //
483 // we also construct the pages menu during this first iteration
484 for ( cat = 0; cat < MAX_PAGES; cat++ )
485 {
486 #if USE_TREEBOOK
487 nPage++; // increase for parent page
488 #else
489 books[cat] = new WidgetsBookCtrl(m_book,
490 wxID_ANY,
491 wxDefaultPosition,
492 wxDefaultSize,
493 wxBK_DEFAULT);
494 #endif
495
496 for ( WidgetsPageInfo *info = WidgetsPage::ms_widgetPages;
497 info;
498 info = info->GetNext() )
499 {
500 if( (info->GetCategories() & ( 1 << cat )) == 0)
501 continue;
502
503 WidgetsPage *page = (*info->GetCtor())(
504 #if USE_TREEBOOK
505 m_book
506 #else
507 books[cat]
508 #endif
509 , imageList);
510 pages[cat].Add(page);
511
512 labels[cat].Add(info->GetLabel());
513 if ( cat == ALL_PAGE )
514 {
515 wxString radioLabel(info->GetLabel());
516 nFKey++;
517 if ( nFKey <= 12 )
518 {
519 radioLabel << wxT("\tF" ) << nFKey;
520 }
521
522 menuPages->AppendRadioItem(
523 Widgets_GoToPage + nPage,
524 radioLabel
525 );
526 #if !USE_TREEBOOK
527 // consider only for book in book architecture
528 nPage++;
529 #endif
530 }
531
532 #if USE_TREEBOOK
533 // consider only for treebook architecture (with subpages)
534 nPage++;
535 #endif
536 }
537 }
538
539 GetMenuBar()->Append(menuPages, _T("&Page"));
540
541 #if USE_ICONS_IN_BOOK
542 m_book->AssignImageList(imageList);
543 #endif
544
545 for ( cat = 0; cat < MAX_PAGES; cat++ )
546 {
547 #if USE_TREEBOOK
548 m_book->AddPage(NULL,WidgetsCategories[cat],false,0);
549 #else
550 m_book->AddPage(books[cat],WidgetsCategories[cat],false,0);
551 #if USE_ICONS_IN_BOOK
552 books[cat]->SetImageList(imageList);
553 #endif
554 #endif
555
556 // now do add them
557 size_t count = pages[cat].GetCount();
558 for ( size_t n = 0; n < count; n++ )
559 {
560 #if USE_TREEBOOK
561 m_book->AddSubPage
562 #else
563 books[cat]->AddPage
564 #endif
565 (
566 pages[cat][n],
567 labels[cat][n],
568 false, // don't select
569 imageId++
570 );
571 }
572 }
573
574 #if USE_TREEBOOK
575 // for treebook page #0 is empty parent page only so select the first page
576 // with some contents
577 m_book->SetSelection(1);
578
579 // but ensure that the top of the tree is shown nevertheless
580 wxTreeCtrl * const tree = m_book->GetTreeCtrl();
581
582 wxTreeItemIdValue cookie;
583 tree->EnsureVisible(tree->GetFirstChild(tree->GetRootItem(), cookie));
584 #endif // USE_TREEBOOK
585 }
586
587 WidgetsPage *WidgetsFrame::CurrentPage()
588 {
589 wxWindow *page = m_book->GetCurrentPage();
590
591 #if !USE_TREEBOOK
592 WidgetsBookCtrl *subBook = wxStaticCast(page, WidgetsBookCtrl);
593 wxCHECK_MSG( subBook, NULL, _T("no WidgetsBookCtrl?") );
594
595 page = subBook->GetCurrentPage();
596 #endif // !USE_TREEBOOK
597
598 return wxStaticCast(page, WidgetsPage);
599 }
600
601 WidgetsFrame::~WidgetsFrame()
602 {
603 #if USE_LOG
604 delete m_logTarget;
605 #endif // USE_LOG
606 }
607
608 // ----------------------------------------------------------------------------
609 // WidgetsFrame event handlers
610 // ----------------------------------------------------------------------------
611
612 void WidgetsFrame::OnExit(wxCommandEvent& WXUNUSED(event))
613 {
614 Close();
615 }
616
617 #if USE_LOG
618 void WidgetsFrame::OnButtonClearLog(wxCommandEvent& WXUNUSED(event))
619 {
620 m_lboxLog->Clear();
621 }
622 #endif // USE_LOG
623
624 #if wxUSE_MENUS
625
626 void WidgetsFrame::OnPageChanging(WidgetsBookCtrlEvent& event)
627 {
628 if ( !m_book->GetPage(event.GetSelection()) )
629 event.Veto();
630 }
631
632 void WidgetsFrame::OnPageChanged(WidgetsBookCtrlEvent& event)
633 {
634 const int sel = event.GetSelection();
635
636 // adjust "Page" menu selection
637 wxMenuItem *item = GetMenuBar()->FindItem(Widgets_GoToPage + sel);
638 if ( item )
639 item->Check();
640
641 GetMenuBar()->Check(Widgets_BusyCursor, false);
642
643 // lazy creation of the pages
644 WidgetsPage *page = CurrentPage();
645 if ( page->GetChildren().empty() )
646 {
647 wxWindowUpdateLocker noUpdates(page);
648 page->CreateContent();
649 WidgetsBookCtrl *book = wxStaticCast(page->GetParent(), WidgetsBookCtrl);
650 wxSize size;
651 for ( size_t i = 0; i < book->GetPageCount(); ++i )
652 {
653 wxWindow *page = book->GetPage(i);
654 if ( page )
655 {
656 size.IncTo(page->GetSize());
657 }
658 }
659 page->SetSize(size);
660 }
661
662 event.Skip();
663 }
664
665 void WidgetsFrame::OnGoToPage(wxCommandEvent& event)
666 {
667 #if USE_TREEBOOK
668 m_book->SetSelection(event.GetId() - Widgets_GoToPage);
669 #else
670 m_book->SetSelection(m_book->GetPageCount()-1);
671 WidgetsBookCtrl *book = wxStaticCast(m_book->GetCurrentPage(), WidgetsBookCtrl);
672 book->SetSelection(event.GetId() - Widgets_GoToPage);
673 #endif
674 }
675
676 #if wxUSE_TOOLTIPS
677
678 void WidgetsFrame::OnSetTooltip(wxCommandEvent& WXUNUSED(event))
679 {
680 static wxString s_tip = _T("This is a tooltip");
681
682 wxTextEntryDialog dialog
683 (
684 this,
685 _T("Tooltip text (may use \\n, leave empty to remove): "),
686 _T("Widgets sample"),
687 s_tip
688 );
689
690 if ( dialog.ShowModal() != wxID_OK )
691 return;
692
693 s_tip = dialog.GetValue();
694 s_tip.Replace(_T("\\n"), _T("\n"));
695
696 WidgetsPage *page = CurrentPage();
697
698 page->GetWidget()->SetToolTip(s_tip);
699
700 wxControl *ctrl2 = page->GetWidget2();
701 if ( ctrl2 )
702 ctrl2->SetToolTip(s_tip);
703 }
704
705 #endif // wxUSE_TOOLTIPS
706
707 void WidgetsFrame::OnSetFgCol(wxCommandEvent& WXUNUSED(event))
708 {
709 #if wxUSE_COLOURDLG
710 // allow for debugging the default colour the first time this is called
711 WidgetsPage *page = CurrentPage();
712
713 if (!m_colFg.Ok())
714 m_colFg = page->GetForegroundColour();
715
716 wxColour col = wxGetColourFromUser(this, m_colFg);
717 if ( !col.Ok() )
718 return;
719
720 m_colFg = col;
721
722 page->GetWidget()->SetForegroundColour(m_colFg);
723 page->GetWidget()->Refresh();
724
725 wxControl *ctrl2 = page->GetWidget2();
726 if ( ctrl2 )
727 {
728 ctrl2->SetForegroundColour(m_colFg);
729 ctrl2->Refresh();
730 }
731 #else
732 wxLogMessage(_T("Colour selection dialog not available in current build."));
733 #endif
734 }
735
736 void WidgetsFrame::OnSetBgCol(wxCommandEvent& WXUNUSED(event))
737 {
738 #if wxUSE_COLOURDLG
739 WidgetsPage *page = CurrentPage();
740
741 if ( !m_colBg.Ok() )
742 m_colBg = page->GetBackgroundColour();
743
744 wxColour col = wxGetColourFromUser(this, m_colBg);
745 if ( !col.Ok() )
746 return;
747
748 m_colBg = col;
749
750 page->GetWidget()->SetBackgroundColour(m_colBg);
751 page->GetWidget()->Refresh();
752
753 wxControl *ctrl2 = page->GetWidget2();
754 if ( ctrl2 )
755 {
756 ctrl2->SetBackgroundColour(m_colFg);
757 ctrl2->Refresh();
758 }
759 #else
760 wxLogMessage(_T("Colour selection dialog not available in current build."));
761 #endif
762 }
763
764 void WidgetsFrame::OnSetFont(wxCommandEvent& WXUNUSED(event))
765 {
766 #if wxUSE_FONTDLG
767 WidgetsPage *page = CurrentPage();
768
769 if (!m_font.Ok())
770 m_font = page->GetFont();
771
772 wxFont font = wxGetFontFromUser(this, m_font);
773 if ( !font.Ok() )
774 return;
775
776 m_font = font;
777
778 page->GetWidget()->SetFont(m_font);
779 page->GetWidget()->Refresh();
780
781 wxControl *ctrl2 = page->GetWidget2();
782 if ( ctrl2 )
783 {
784 ctrl2->SetFont(m_font);
785 ctrl2->Refresh();
786 }
787 #else
788 wxLogMessage(_T("Font selection dialog not available in current build."));
789 #endif
790 }
791
792 void WidgetsFrame::OnEnable(wxCommandEvent& event)
793 {
794 CurrentPage()->GetWidget()->Enable(event.IsChecked());
795 }
796
797 void WidgetsFrame::OnSetBorder(wxCommandEvent& event)
798 {
799 int border;
800 switch ( event.GetId() )
801 {
802 case Widgets_BorderNone: border = wxBORDER_NONE; break;
803 case Widgets_BorderStatic: border = wxBORDER_STATIC; break;
804 case Widgets_BorderSimple: border = wxBORDER_SIMPLE; break;
805 case Widgets_BorderRaised: border = wxBORDER_RAISED; break;
806 case Widgets_BorderSunken: border = wxBORDER_SUNKEN; break;
807 case Widgets_BorderDouble: border = wxBORDER_DOUBLE; break;
808
809 default:
810 wxFAIL_MSG( _T("unknown border style") );
811 // fall through
812
813 case Widgets_BorderDefault: border = wxBORDER_DEFAULT; break;
814 }
815
816 WidgetsPage::ms_defaultFlags &= ~wxBORDER_MASK;
817 WidgetsPage::ms_defaultFlags |= border;
818
819 WidgetsPage *page = CurrentPage();
820
821 page->RecreateWidget();
822 }
823
824 void WidgetsFrame::OnToggleGlobalBusyCursor(wxCommandEvent& event)
825 {
826 if ( event.IsChecked() )
827 wxBeginBusyCursor();
828 else
829 wxEndBusyCursor();
830 }
831
832 void WidgetsFrame::OnToggleBusyCursor(wxCommandEvent& event)
833 {
834 CurrentPage()->GetWidget()->SetCursor(*(event.IsChecked()
835 ? wxHOURGLASS_CURSOR
836 : wxSTANDARD_CURSOR));
837 }
838
839 #endif // wxUSE_MENUS
840
841 // ----------------------------------------------------------------------------
842 // WidgetsPageInfo
843 // ----------------------------------------------------------------------------
844
845 WidgetsPageInfo::WidgetsPageInfo(Constructor ctor, const wxChar *label, int categories)
846 : m_label(label)
847 , m_categories(categories)
848 {
849 m_ctor = ctor;
850
851 m_next = NULL;
852
853 // dummy sorting: add and immediately sort in the list according to label
854 if ( WidgetsPage::ms_widgetPages )
855 {
856 WidgetsPageInfo *node_prev = WidgetsPage::ms_widgetPages;
857 if ( wxStrcmp(label, node_prev->GetLabel().c_str()) < 0 )
858 {
859 // add as first
860 m_next = node_prev;
861 WidgetsPage::ms_widgetPages = this;
862 }
863 else
864 {
865 WidgetsPageInfo *node_next;
866 do
867 {
868 node_next = node_prev->GetNext();
869 if ( node_next )
870 {
871 // add if between two
872 if ( wxStrcmp(label, node_next->GetLabel().c_str()) < 0 )
873 {
874 node_prev->SetNext(this);
875 m_next = node_next;
876 // force to break loop
877 node_next = NULL;
878 }
879 }
880 else
881 {
882 // add as last
883 node_prev->SetNext(this);
884 m_next = node_next;
885 }
886 node_prev = node_next;
887 }
888 while ( node_next );
889 }
890 }
891 else
892 {
893 // add when first
894 WidgetsPage::ms_widgetPages = this;
895 }
896 }
897
898 // ----------------------------------------------------------------------------
899 // WidgetsPage
900 // ----------------------------------------------------------------------------
901
902 int WidgetsPage::ms_defaultFlags = wxBORDER_DEFAULT;
903 WidgetsPageInfo *WidgetsPage::ms_widgetPages = NULL;
904
905 WidgetsPage::WidgetsPage(WidgetsBookCtrl *book,
906 wxImageList *imaglist,
907 char* icon[])
908 : wxPanel(book, wxID_ANY,
909 wxDefaultPosition, wxDefaultSize,
910 wxNO_FULL_REPAINT_ON_RESIZE |
911 wxCLIP_CHILDREN |
912 wxTAB_TRAVERSAL)
913 {
914 #if USE_ICONS_IN_BOOK
915 imaglist->Add(wxBitmap(icon));
916 #else
917 wxUnusedVar(imaglist);
918 wxUnusedVar(icon);
919 #endif
920 }
921
922 wxSizer *WidgetsPage::CreateSizerWithText(wxControl *control,
923 wxWindowID id,
924 wxTextCtrl **ppText)
925 {
926 wxSizer *sizerRow = new wxBoxSizer(wxHORIZONTAL);
927 wxTextCtrl *text = new wxTextCtrl(this, id, wxEmptyString,
928 wxDefaultPosition, wxDefaultSize, wxTE_PROCESS_ENTER);
929
930 sizerRow->Add(control, 0, wxRIGHT | wxALIGN_CENTRE_VERTICAL, 5);
931 sizerRow->Add(text, 1, wxLEFT | wxALIGN_CENTRE_VERTICAL, 5);
932
933 if ( ppText )
934 *ppText = text;
935
936 return sizerRow;
937 }
938
939 // create a sizer containing a label and a text ctrl
940 wxSizer *WidgetsPage::CreateSizerWithTextAndLabel(const wxString& label,
941 wxWindowID id,
942 wxTextCtrl **ppText)
943 {
944 return CreateSizerWithText(new wxStaticText(this, wxID_ANY, label),
945 id, ppText);
946 }
947
948 // create a sizer containing a button and a text ctrl
949 wxSizer *WidgetsPage::CreateSizerWithTextAndButton(wxWindowID idBtn,
950 const wxString& label,
951 wxWindowID id,
952 wxTextCtrl **ppText)
953 {
954 return CreateSizerWithText(new wxButton(this, idBtn, label), id, ppText);
955 }
956
957 wxCheckBox *WidgetsPage::CreateCheckBoxAndAddToSizer(wxSizer *sizer,
958 const wxString& label,
959 wxWindowID id)
960 {
961 wxCheckBox *checkbox = new wxCheckBox(this, id, label);
962 sizer->Add(checkbox, 0, wxLEFT | wxRIGHT, 5);
963 sizer->Add(0, 2, 0, wxGROW); // spacer
964
965 return checkbox;
966 }