1426 lines
		
	
	
		
			42 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1426 lines
		
	
	
		
			42 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /////////////////////////////////////////////////////////////////////////////
 | |
| // Program:     wxWidgets Widgets Sample
 | |
| // Name:        samples/widgets/widgets.cpp
 | |
| // Purpose:     Sample showing most of the simple wxWidgets widgets
 | |
| // Author:      Vadim Zeitlin
 | |
| // Created:     27.03.01
 | |
| // Copyright:   (c) 2001 Vadim Zeitlin
 | |
| // Licence:     wxWindows licence
 | |
| /////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| // ============================================================================
 | |
| // declarations
 | |
| // ============================================================================
 | |
| 
 | |
| // ----------------------------------------------------------------------------
 | |
| // headers
 | |
| // ----------------------------------------------------------------------------
 | |
| 
 | |
| // for compilers that support precompilation, includes "wx/wx.h".
 | |
| #include "wx/wxprec.h"
 | |
| 
 | |
| 
 | |
| // for all others, include the necessary headers
 | |
| #ifndef WX_PRECOMP
 | |
|     #include "wx/app.h"
 | |
|     #include "wx/log.h"
 | |
|     #include "wx/frame.h"
 | |
|     #include "wx/menu.h"
 | |
|     #include "wx/image.h"
 | |
| 
 | |
|     #include "wx/button.h"
 | |
|     #include "wx/checkbox.h"
 | |
|     #include "wx/listbox.h"
 | |
|     #include "wx/statbox.h"
 | |
|     #include "wx/stattext.h"
 | |
|     #include "wx/textctrl.h"
 | |
|     #include "wx/msgdlg.h"
 | |
| #endif
 | |
| 
 | |
| #include "wx/sysopt.h"
 | |
| #include "wx/bookctrl.h"
 | |
| #include "wx/treebook.h"
 | |
| #include "wx/sizer.h"
 | |
| #include "wx/colordlg.h"
 | |
| #include "wx/fontdlg.h"
 | |
| #include "wx/numdlg.h"
 | |
| #include "wx/textdlg.h"
 | |
| #include "wx/imaglist.h"
 | |
| #include "wx/treectrl.h"
 | |
| #include "wx/wupdlock.h"
 | |
| #include "wx/textcompleter.h"
 | |
| 
 | |
| #include "wx/persist/toplevel.h"
 | |
| #include "wx/persist/treebook.h"
 | |
| 
 | |
| #include "widgets.h"
 | |
| 
 | |
| #include "../sample.xpm"
 | |
| 
 | |
| // ----------------------------------------------------------------------------
 | |
| // constants
 | |
| // ----------------------------------------------------------------------------
 | |
| 
 | |
| // control ids
 | |
| enum
 | |
| {
 | |
|     Widgets_ClearLog = 100,
 | |
|     Widgets_Quit,
 | |
| 
 | |
|     Widgets_BookCtrl,
 | |
| 
 | |
| #if wxUSE_TOOLTIPS
 | |
|     Widgets_SetTooltip,
 | |
| #endif // wxUSE_TOOLTIPS
 | |
|     Widgets_SetFgColour,
 | |
|     Widgets_SetBgColour,
 | |
|     Widgets_SetPageBg,
 | |
|     Widgets_SetFont,
 | |
|     Widgets_Enable,
 | |
|     Widgets_Show,
 | |
| 
 | |
|     Widgets_BorderNone,
 | |
|     Widgets_BorderStatic,
 | |
|     Widgets_BorderSimple,
 | |
|     Widgets_BorderRaised,
 | |
|     Widgets_BorderSunken,
 | |
|     Widgets_BorderDouble,
 | |
|     Widgets_BorderDefault,
 | |
| 
 | |
|     Widgets_VariantNormal,
 | |
|     Widgets_VariantSmall,
 | |
|     Widgets_VariantMini,
 | |
|     Widgets_VariantLarge,
 | |
| 
 | |
|     Widgets_LayoutDirection,
 | |
| 
 | |
|     Widgets_GlobalBusyCursor,
 | |
|     Widgets_BusyCursor,
 | |
| 
 | |
|     Widgets_GoToPage,
 | |
|     Widgets_GoToPageLast = Widgets_GoToPage + 100,
 | |
| 
 | |
| 
 | |
|     TextEntry_Begin,
 | |
|     TextEntry_DisableAutoComplete = TextEntry_Begin,
 | |
|     TextEntry_AutoCompleteFixed,
 | |
|     TextEntry_AutoCompleteFilenames,
 | |
|     TextEntry_AutoCompleteDirectories,
 | |
|     TextEntry_AutoCompleteCustom,
 | |
|     TextEntry_AutoCompleteKeyLength,
 | |
| 
 | |
|     TextEntry_SetHint,
 | |
|     TextEntry_End
 | |
| };
 | |
| 
 | |
| const wxChar *WidgetsCategories[MAX_PAGES] = {
 | |
| #if defined(__WXUNIVERSAL__)
 | |
|     wxT("Universal"),
 | |
| #else
 | |
|     wxT("Native"),
 | |
| #endif
 | |
|     wxT("Generic"),
 | |
|     wxT("Pickers"),
 | |
|     wxT("Comboboxes"),
 | |
|     wxT("With items"),
 | |
|     wxT("Editable"),
 | |
|     wxT("Books"),
 | |
|     wxT("All controls")
 | |
| };
 | |
| 
 | |
| // ----------------------------------------------------------------------------
 | |
| // our classes
 | |
| // ----------------------------------------------------------------------------
 | |
| 
 | |
| // Define a new application type, each program should derive a class from wxApp
 | |
| class WidgetsApp : public wxApp
 | |
| {
 | |
| public:
 | |
|     WidgetsApp()
 | |
|     {
 | |
| #if USE_LOG
 | |
|         m_logTarget = NULL;
 | |
| #endif // USE_LOG
 | |
|     }
 | |
| 
 | |
|     // override base class virtuals
 | |
|     // ----------------------------
 | |
| 
 | |
|     // this one is called on application startup and is a good place for the app
 | |
|     // initialization (doing it here and not in the ctor allows to have an error
 | |
|     // return: if OnInit() returns false, the application terminates)
 | |
|     virtual bool OnInit() wxOVERRIDE;
 | |
| 
 | |
|     // real implementation of WidgetsPage method with the same name
 | |
|     bool IsUsingLogWindow() const;
 | |
| 
 | |
| private:
 | |
| #if USE_LOG
 | |
|     wxLog* m_logTarget;
 | |
| #endif // USE_LOG
 | |
| 
 | |
|     wxDECLARE_NO_COPY_CLASS(WidgetsApp);
 | |
| };
 | |
| 
 | |
| wxDECLARE_APP(WidgetsApp); // This provides a convenient wxGetApp() accessor.
 | |
| 
 | |
| // Define a new frame type: this is going to be our main frame
 | |
| class WidgetsFrame : public wxFrame
 | |
| {
 | |
| public:
 | |
|     // ctor(s) and dtor
 | |
|     WidgetsFrame(const wxString& title);
 | |
|     virtual ~WidgetsFrame();
 | |
| 
 | |
| protected:
 | |
|     // event handlers
 | |
| #if USE_LOG
 | |
|     void OnButtonClearLog(wxCommandEvent& event);
 | |
| #endif // USE_LOG
 | |
|     void OnExit(wxCommandEvent& event);
 | |
| 
 | |
| #if wxUSE_MENUS
 | |
|     void OnPageChanging(WidgetsBookCtrlEvent& event);
 | |
|     void OnPageChanged(WidgetsBookCtrlEvent& event);
 | |
|     void OnGoToPage(wxCommandEvent& event);
 | |
| 
 | |
| #if wxUSE_TOOLTIPS
 | |
|     void OnSetTooltip(wxCommandEvent& event);
 | |
| #endif // wxUSE_TOOLTIPS
 | |
|     void OnSetFgCol(wxCommandEvent& event);
 | |
|     void OnSetBgCol(wxCommandEvent& event);
 | |
|     void OnSetPageBg(wxCommandEvent& event);
 | |
|     void OnSetFont(wxCommandEvent& event);
 | |
|     void OnEnable(wxCommandEvent& event);
 | |
|     void OnShow(wxCommandEvent &event);
 | |
|     void OnSetBorder(wxCommandEvent& event);
 | |
|     void OnSetVariant(wxCommandEvent& event);
 | |
| 
 | |
|     void OnToggleLayoutDirection(wxCommandEvent& event);
 | |
| 
 | |
|     void OnToggleGlobalBusyCursor(wxCommandEvent& event);
 | |
|     void OnToggleBusyCursor(wxCommandEvent& event);
 | |
| 
 | |
|     // wxTextEntry-specific tests
 | |
|     void OnDisableAutoComplete(wxCommandEvent& event);
 | |
|     void OnAutoCompleteFixed(wxCommandEvent& event);
 | |
|     void OnAutoCompleteFilenames(wxCommandEvent& event);
 | |
|     void OnAutoCompleteDirectories(wxCommandEvent& event);
 | |
|     void OnAutoCompleteCustom(wxCommandEvent& event);
 | |
|     void OnAutoCompleteKeyLength(wxCommandEvent& event);
 | |
| 
 | |
|     void DoUseCustomAutoComplete(size_t minLength = 1);
 | |
| 
 | |
|     void OnSetHint(wxCommandEvent& event);
 | |
| 
 | |
|     void OnUpdateTextUI(wxUpdateUIEvent& event)
 | |
|     {
 | |
|         event.Enable( CurrentPage()->GetTextEntry() != NULL );
 | |
|     }
 | |
| #endif // wxUSE_MENUS
 | |
| 
 | |
|     // initialize the book: add all pages to it
 | |
|     void InitBook();
 | |
| 
 | |
|     // return the currently selected page (never NULL)
 | |
|     WidgetsPage *CurrentPage();
 | |
| 
 | |
| private:
 | |
|     void OnWidgetFocus(wxFocusEvent& event);
 | |
| 
 | |
|     void ConnectToWidgetEvents();
 | |
| 
 | |
|     // the panel containing everything
 | |
|     wxPanel *m_panel;
 | |
| 
 | |
| #if USE_LOG
 | |
|     // the listbox for logging messages
 | |
|     wxListBox *m_lboxLog;
 | |
| 
 | |
|     // the log target we use to redirect messages to the listbox
 | |
|     wxLog *m_logTarget;
 | |
| #endif // USE_LOG
 | |
| 
 | |
|     // the book containing the test pages
 | |
|     WidgetsBookCtrl *m_book;
 | |
| 
 | |
|     // any class wishing to process wxWidgets events must use this macro
 | |
|     wxDECLARE_EVENT_TABLE();
 | |
| };
 | |
| 
 | |
| #if USE_LOG
 | |
| // A log target which just redirects the messages to a listbox
 | |
| class LboxLogger : public wxLog
 | |
| {
 | |
| public:
 | |
|     LboxLogger(wxListBox *lbox, wxLog *logOld)
 | |
|     {
 | |
|         m_lbox = lbox;
 | |
|         //m_lbox->Disable(); -- looks ugly under MSW
 | |
|         m_logOld = logOld;
 | |
|     }
 | |
| 
 | |
|     virtual ~LboxLogger()
 | |
|     {
 | |
|         wxLog::SetActiveTarget(m_logOld);
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     // implement sink functions
 | |
|     virtual void DoLogTextAtLevel(wxLogLevel level, const wxString& msg) wxOVERRIDE
 | |
|     {
 | |
|         if ( level == wxLOG_Trace )
 | |
|         {
 | |
|             if ( m_logOld )
 | |
|                 m_logOld->LogTextAtLevel(level, msg);
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         #ifdef __WXUNIVERSAL__
 | |
|             m_lbox->AppendAndEnsureVisible(msg);
 | |
|         #else // other ports don't have this method yet
 | |
|             m_lbox->Append(msg);
 | |
|             m_lbox->SetFirstItem(m_lbox->GetCount() - 1);
 | |
|         #endif
 | |
|     }
 | |
| 
 | |
|     // the control we use
 | |
|     wxListBox *m_lbox;
 | |
| 
 | |
|     // the old log target
 | |
|     wxLog *m_logOld;
 | |
| };
 | |
| #endif // USE_LOG
 | |
| 
 | |
| // array of pages
 | |
| WX_DEFINE_ARRAY_PTR(WidgetsPage *, ArrayWidgetsPage);
 | |
| 
 | |
| // ----------------------------------------------------------------------------
 | |
| // misc macros
 | |
| // ----------------------------------------------------------------------------
 | |
| 
 | |
| wxIMPLEMENT_APP(WidgetsApp);
 | |
| 
 | |
| // ----------------------------------------------------------------------------
 | |
| // event tables
 | |
| // ----------------------------------------------------------------------------
 | |
| 
 | |
| wxBEGIN_EVENT_TABLE(WidgetsFrame, wxFrame)
 | |
| #if USE_LOG
 | |
|     EVT_BUTTON(Widgets_ClearLog, WidgetsFrame::OnButtonClearLog)
 | |
| #endif // USE_LOG
 | |
|     EVT_BUTTON(Widgets_Quit, WidgetsFrame::OnExit)
 | |
| 
 | |
| #if wxUSE_TOOLTIPS
 | |
|     EVT_MENU(Widgets_SetTooltip, WidgetsFrame::OnSetTooltip)
 | |
| #endif // wxUSE_TOOLTIPS
 | |
| 
 | |
| #if wxUSE_MENUS
 | |
|     EVT_WIDGETS_PAGE_CHANGING(wxID_ANY, WidgetsFrame::OnPageChanging)
 | |
|     EVT_MENU_RANGE(Widgets_GoToPage, Widgets_GoToPageLast,
 | |
|                    WidgetsFrame::OnGoToPage)
 | |
| 
 | |
|     EVT_MENU(Widgets_SetFgColour, WidgetsFrame::OnSetFgCol)
 | |
|     EVT_MENU(Widgets_SetBgColour, WidgetsFrame::OnSetBgCol)
 | |
|     EVT_MENU(Widgets_SetPageBg,   WidgetsFrame::OnSetPageBg)
 | |
|     EVT_MENU(Widgets_SetFont,     WidgetsFrame::OnSetFont)
 | |
|     EVT_MENU(Widgets_Enable,      WidgetsFrame::OnEnable)
 | |
|     EVT_MENU(Widgets_Show,        WidgetsFrame::OnShow)
 | |
| 
 | |
|     EVT_MENU_RANGE(Widgets_BorderNone, Widgets_BorderDefault,
 | |
|                    WidgetsFrame::OnSetBorder)
 | |
| 
 | |
|     EVT_MENU_RANGE(Widgets_VariantNormal, Widgets_VariantLarge,
 | |
|                    WidgetsFrame::OnSetVariant)
 | |
| 
 | |
|     EVT_MENU(Widgets_LayoutDirection,   WidgetsFrame::OnToggleLayoutDirection)
 | |
| 
 | |
|     EVT_MENU(Widgets_GlobalBusyCursor,  WidgetsFrame::OnToggleGlobalBusyCursor)
 | |
|     EVT_MENU(Widgets_BusyCursor,        WidgetsFrame::OnToggleBusyCursor)
 | |
| 
 | |
|     EVT_MENU(TextEntry_DisableAutoComplete,   WidgetsFrame::OnDisableAutoComplete)
 | |
|     EVT_MENU(TextEntry_AutoCompleteFixed,     WidgetsFrame::OnAutoCompleteFixed)
 | |
|     EVT_MENU(TextEntry_AutoCompleteFilenames, WidgetsFrame::OnAutoCompleteFilenames)
 | |
|     EVT_MENU(TextEntry_AutoCompleteDirectories, WidgetsFrame::OnAutoCompleteDirectories)
 | |
|     EVT_MENU(TextEntry_AutoCompleteCustom,    WidgetsFrame::OnAutoCompleteCustom)
 | |
|     EVT_MENU(TextEntry_AutoCompleteKeyLength, WidgetsFrame::OnAutoCompleteKeyLength)
 | |
| 
 | |
|     EVT_MENU(TextEntry_SetHint, WidgetsFrame::OnSetHint)
 | |
| 
 | |
|     EVT_UPDATE_UI_RANGE(TextEntry_Begin, TextEntry_End - 1,
 | |
|                         WidgetsFrame::OnUpdateTextUI)
 | |
| 
 | |
|     EVT_MENU(wxID_EXIT, WidgetsFrame::OnExit)
 | |
| #endif // wxUSE_MENUS
 | |
| wxEND_EVENT_TABLE()
 | |
| 
 | |
| // ============================================================================
 | |
| // implementation
 | |
| // ============================================================================
 | |
| 
 | |
| // ----------------------------------------------------------------------------
 | |
| // app class
 | |
| // ----------------------------------------------------------------------------
 | |
| 
 | |
| bool WidgetsApp::OnInit()
 | |
| {
 | |
|     if ( !wxApp::OnInit() )
 | |
|         return false;
 | |
| 
 | |
|     SetVendorName("wxWidgets_Samples");
 | |
| 
 | |
|     // the reason for having these ifdef's is that I often run two copies of
 | |
|     // this sample side by side and it is useful to see which one is which
 | |
|     wxString title;
 | |
| #if defined(__WXUNIVERSAL__)
 | |
|     title = "wxUniv/";
 | |
| #endif
 | |
| 
 | |
| #if defined(__WXMSW__)
 | |
|     title += "wxMSW";
 | |
| #elif defined(__WXGTK__)
 | |
|     title += "wxGTK";
 | |
| #elif defined(__WXMAC__)
 | |
|     title += "wxMAC";
 | |
| #elif defined(__WXMOTIF__)
 | |
|     title += "wxMOTIF";
 | |
| #else
 | |
|     title += "wxWidgets";
 | |
| #endif
 | |
| 
 | |
|     wxFrame *frame = new WidgetsFrame(title + " widgets demo");
 | |
|     frame->Show();
 | |
| 
 | |
| #if USE_LOG
 | |
|     m_logTarget = wxLog::GetActiveTarget();
 | |
| #endif // USE_LOG
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| bool WidgetsApp::IsUsingLogWindow() const
 | |
| {
 | |
| #if USE_LOG
 | |
|     return wxLog::GetActiveTarget() == m_logTarget;
 | |
| #else // !USE_LOG
 | |
|     return false;
 | |
| #endif // USE_LOG
 | |
| }
 | |
| 
 | |
| // ----------------------------------------------------------------------------
 | |
| // WidgetsFrame construction
 | |
| // ----------------------------------------------------------------------------
 | |
| 
 | |
| WidgetsFrame::WidgetsFrame(const wxString& title)
 | |
|             : wxFrame(NULL, wxID_ANY, title)
 | |
| {
 | |
|     // set the frame icon
 | |
|     SetIcon(wxICON(sample));
 | |
| 
 | |
|     // init everything
 | |
| #if USE_LOG
 | |
|     m_lboxLog = NULL;
 | |
|     m_logTarget = NULL;
 | |
| #endif // USE_LOG
 | |
|     m_book = NULL;
 | |
| 
 | |
| #if wxUSE_MENUS
 | |
|     // create the menubar
 | |
|     wxMenuBar *mbar = new wxMenuBar;
 | |
|     wxMenu *menuWidget = new wxMenu;
 | |
| #if wxUSE_TOOLTIPS
 | |
|     menuWidget->Append(Widgets_SetTooltip, "Set &tooltip...\tCtrl-T");
 | |
|     menuWidget->AppendSeparator();
 | |
| #endif // wxUSE_TOOLTIPS
 | |
|     menuWidget->Append(Widgets_SetFgColour, "Set &foreground...\tCtrl-F");
 | |
|     menuWidget->Append(Widgets_SetBgColour, "Set &background...\tCtrl-B");
 | |
|     menuWidget->Append(Widgets_SetPageBg,   "Set &page background...\tShift-Ctrl-B");
 | |
|     menuWidget->Append(Widgets_SetFont,     "Set f&ont...\tCtrl-O");
 | |
|     menuWidget->AppendCheckItem(Widgets_Enable,  "&Enable/disable\tCtrl-E");
 | |
|     menuWidget->AppendCheckItem(Widgets_Show, "Show/Hide");
 | |
| 
 | |
|     wxMenu *menuBorders = new wxMenu;
 | |
|     menuBorders->AppendRadioItem(Widgets_BorderDefault, "De&fault\tCtrl-Shift-9");
 | |
|     menuBorders->AppendRadioItem(Widgets_BorderNone,   "&None\tCtrl-Shift-0");
 | |
|     menuBorders->AppendRadioItem(Widgets_BorderSimple, "&Simple\tCtrl-Shift-1");
 | |
|     menuBorders->AppendRadioItem(Widgets_BorderDouble, "&Double\tCtrl-Shift-2");
 | |
|     menuBorders->AppendRadioItem(Widgets_BorderStatic, "Stati&c\tCtrl-Shift-3");
 | |
|     menuBorders->AppendRadioItem(Widgets_BorderRaised, "&Raised\tCtrl-Shift-4");
 | |
|     menuBorders->AppendRadioItem(Widgets_BorderSunken, "S&unken\tCtrl-Shift-5");
 | |
|     menuWidget->AppendSubMenu(menuBorders, "Set &border");
 | |
| 
 | |
|     wxMenu* const menuVariants = new wxMenu;
 | |
|     menuVariants->AppendRadioItem(Widgets_VariantMini, "&Mini\tCtrl-Shift-6");
 | |
|     menuVariants->AppendRadioItem(Widgets_VariantSmall, "&Small\tCtrl-Shift-7");
 | |
|     menuVariants->AppendRadioItem(Widgets_VariantNormal, "&Normal\tCtrl-Shift-8");
 | |
|     menuVariants->AppendRadioItem(Widgets_VariantLarge, "&Large\tCtrl-Shift-9");
 | |
|     menuWidget->AppendSubMenu(menuVariants, "Set &variant");
 | |
| 
 | |
|     menuWidget->AppendSeparator();
 | |
|     menuWidget->AppendCheckItem(Widgets_LayoutDirection,
 | |
|                                 "Toggle &layout direction\tCtrl-L");
 | |
|     menuWidget->Check(Widgets_LayoutDirection,
 | |
|                       GetLayoutDirection() == wxLayout_RightToLeft);
 | |
| 
 | |
|     menuWidget->AppendSeparator();
 | |
|     menuWidget->AppendCheckItem(Widgets_GlobalBusyCursor,
 | |
|                                 "Toggle &global busy cursor\tCtrl-Shift-U");
 | |
|     menuWidget->AppendCheckItem(Widgets_BusyCursor,
 | |
|                                 "Toggle b&usy cursor\tCtrl-U");
 | |
| 
 | |
|     menuWidget->AppendSeparator();
 | |
|     menuWidget->Append(wxID_EXIT, "&Quit\tCtrl-Q");
 | |
|     mbar->Append(menuWidget, "&Widget");
 | |
| 
 | |
|     wxMenu *menuTextEntry = new wxMenu;
 | |
|     menuTextEntry->AppendRadioItem(TextEntry_DisableAutoComplete,
 | |
|                                    "&Disable auto-completion");
 | |
|     menuTextEntry->AppendRadioItem(TextEntry_AutoCompleteFixed,
 | |
|                                    "Fixed-&list auto-completion");
 | |
|     menuTextEntry->AppendRadioItem(TextEntry_AutoCompleteFilenames,
 | |
|                                    "&Files names auto-completion");
 | |
|     menuTextEntry->AppendRadioItem(TextEntry_AutoCompleteDirectories,
 | |
|                                    "&Directories names auto-completion");
 | |
|     menuTextEntry->AppendRadioItem(TextEntry_AutoCompleteCustom,
 | |
|                                    "&Custom auto-completion");
 | |
|     menuTextEntry->AppendRadioItem(TextEntry_AutoCompleteKeyLength,
 | |
|                                    "Custom with &min length");
 | |
|     menuTextEntry->AppendSeparator();
 | |
|     menuTextEntry->Append(TextEntry_SetHint, "Set help &hint");
 | |
| 
 | |
|     mbar->Append(menuTextEntry, "&Text");
 | |
| 
 | |
|     SetMenuBar(mbar);
 | |
| 
 | |
|     mbar->Check(Widgets_Enable, true);
 | |
|     mbar->Check(Widgets_Show, true);
 | |
| 
 | |
|     mbar->Check(Widgets_VariantNormal, true);
 | |
| #endif // wxUSE_MENUS
 | |
| 
 | |
|     // create controls
 | |
|     m_panel = new wxPanel(this, wxID_ANY);
 | |
| 
 | |
|     wxSizer *sizerTop = new wxBoxSizer(wxVERTICAL);
 | |
| 
 | |
|     // we have 2 panes: book with pages demonstrating the controls in the
 | |
|     // upper one and the log window with some buttons in the lower
 | |
| 
 | |
|     int style = wxBK_DEFAULT;
 | |
|     // Uncomment to suppress page theme (draw in solid colour)
 | |
|     //style |= wxNB_NOPAGETHEME;
 | |
| 
 | |
|     m_book = new WidgetsBookCtrl(m_panel, Widgets_BookCtrl,
 | |
|                                  wxDefaultPosition, wxDefaultSize,
 | |
|                                  style, "Widgets");
 | |
| 
 | |
|     InitBook();
 | |
| 
 | |
|     // the lower one only has the log listbox and a button to clear it
 | |
| #if USE_LOG
 | |
|     wxSizer *sizerDown = new wxStaticBoxSizer(
 | |
|         new wxStaticBox( m_panel, wxID_ANY, "&Log window" ),
 | |
|         wxVERTICAL);
 | |
| 
 | |
|     m_lboxLog = new wxListBox(m_panel, wxID_ANY);
 | |
|     sizerDown->Add(m_lboxLog, wxSizerFlags(1).Expand().Border());
 | |
|     sizerDown->SetMinSize(100, 150);
 | |
| #else
 | |
|     wxSizer *sizerDown = new wxBoxSizer(wxVERTICAL);
 | |
| #endif // USE_LOG
 | |
| 
 | |
|     wxBoxSizer *sizerBtns = new wxBoxSizer(wxHORIZONTAL);
 | |
|     wxButton *btn;
 | |
| #if USE_LOG
 | |
|     btn = new wxButton(m_panel, Widgets_ClearLog, "Clear &log");
 | |
|     sizerBtns->Add(btn);
 | |
|     sizerBtns->AddSpacer(10);
 | |
| #endif // USE_LOG
 | |
|     btn = new wxButton(m_panel, Widgets_Quit, "E&xit");
 | |
|     sizerBtns->Add(btn);
 | |
|     sizerDown->Add(sizerBtns, wxSizerFlags().Border().Right());
 | |
| 
 | |
|     // put everything together
 | |
|     sizerTop->Add(m_book, wxSizerFlags(1).Expand().DoubleBorder(wxALL & ~(wxTOP | wxBOTTOM)));
 | |
|     sizerTop->AddSpacer(5);
 | |
|     sizerTop->Add(sizerDown, wxSizerFlags(0).Expand().DoubleBorder(wxALL & ~wxTOP));
 | |
| 
 | |
|     m_panel->SetSizer(sizerTop);
 | |
| 
 | |
|     const bool sizeSet = wxPersistentRegisterAndRestore(this, "Main");
 | |
| 
 | |
|     const wxSize sizeMin = m_panel->GetBestSize();
 | |
|     if ( !sizeSet )
 | |
|         SetClientSize(sizeMin);
 | |
|     SetMinClientSize(sizeMin);
 | |
| 
 | |
| #if USE_LOG
 | |
|     // now that everything is created we can redirect the log messages to the
 | |
|     // listbox
 | |
|     m_logTarget = new LboxLogger(m_lboxLog, wxLog::GetActiveTarget());
 | |
|     wxLog::SetActiveTarget(m_logTarget);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::InitBook()
 | |
| {
 | |
|     wxImageList *imageList = new wxImageList(ICON_SIZE, ICON_SIZE);
 | |
| 
 | |
|     wxImage img(sample_xpm);
 | |
|     imageList->Add(wxBitmap(img.Scale(ICON_SIZE, ICON_SIZE)));
 | |
| 
 | |
| #if !USE_TREEBOOK
 | |
|     WidgetsBookCtrl *books[MAX_PAGES];
 | |
| #endif
 | |
| 
 | |
|     ArrayWidgetsPage pages[MAX_PAGES];
 | |
|     wxArrayString labels[MAX_PAGES];
 | |
| 
 | |
|     wxMenu *menuPages = new wxMenu;
 | |
|     unsigned int nPage = 0;
 | |
|     int cat, imageId = 1;
 | |
| 
 | |
|     // we need to first create all pages and only then add them to the book
 | |
|     // as we need the image list first
 | |
|     //
 | |
|     // we also construct the pages menu during this first iteration
 | |
|     for ( cat = 0; cat < MAX_PAGES; cat++ )
 | |
|     {
 | |
| #if USE_TREEBOOK
 | |
|         nPage++; // increase for parent page
 | |
| #else
 | |
|         books[cat] = new WidgetsBookCtrl(m_book,
 | |
|                                          wxID_ANY,
 | |
|                                          wxDefaultPosition,
 | |
|                                          wxDefaultSize,
 | |
|                                          wxBK_DEFAULT);
 | |
| #endif
 | |
| 
 | |
|         for ( WidgetsPageInfo *info = WidgetsPage::ms_widgetPages;
 | |
|               info;
 | |
|               info = info->GetNext() )
 | |
|         {
 | |
|             if( (info->GetCategories() & ( 1 << cat )) == 0)
 | |
|                 continue;
 | |
| 
 | |
|             WidgetsPage *page = (*info->GetCtor())(
 | |
| #if USE_TREEBOOK
 | |
|                                  m_book
 | |
| #else
 | |
|                                  books[cat]
 | |
| #endif
 | |
|                                  , imageList);
 | |
|             pages[cat].Add(page);
 | |
| 
 | |
|             labels[cat].Add(info->GetLabel());
 | |
|             if ( cat == ALL_PAGE )
 | |
|             {
 | |
|                 menuPages->AppendRadioItem(
 | |
|                             Widgets_GoToPage + nPage,
 | |
|                             info->GetLabel()
 | |
|                            );
 | |
| #if !USE_TREEBOOK
 | |
|                 // consider only for book in book architecture
 | |
|                 nPage++;
 | |
| #endif
 | |
|             }
 | |
| 
 | |
| #if USE_TREEBOOK
 | |
|             // consider only for treebook architecture (with subpages)
 | |
|             nPage++;
 | |
| #endif
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     GetMenuBar()->Append(menuPages, "&Page");
 | |
| 
 | |
|     m_book->AssignImageList(imageList);
 | |
| 
 | |
|     for ( cat = 0; cat < MAX_PAGES; cat++ )
 | |
|     {
 | |
| #if USE_TREEBOOK
 | |
|         m_book->AddPage(NULL,WidgetsCategories[cat],false,0);
 | |
| #else
 | |
|         m_book->AddPage(books[cat],WidgetsCategories[cat],false,0);
 | |
|         books[cat]->SetImageList(imageList);
 | |
| #endif
 | |
| 
 | |
|         // now do add them
 | |
|         size_t count = pages[cat].GetCount();
 | |
|         for ( size_t n = 0; n < count; n++ )
 | |
|         {
 | |
| #if USE_TREEBOOK
 | |
|             m_book->AddSubPage
 | |
| #else
 | |
|             books[cat]->AddPage
 | |
| #endif
 | |
|                            (
 | |
|                             pages[cat][n],
 | |
|                             labels[cat][n],
 | |
|                             false, // don't select
 | |
|                             imageId++
 | |
|                            );
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     Bind(wxEVT_COMMAND_WIDGETS_PAGE_CHANGED, &WidgetsFrame::OnPageChanged, this);
 | |
| 
 | |
|     const bool pageSet = wxPersistentRegisterAndRestore(m_book);
 | |
| 
 | |
| #if USE_TREEBOOK
 | |
|     // for treebook page #0 is empty parent page only so select the first page
 | |
|     // with some contents
 | |
|     if ( !pageSet || !m_book->GetCurrentPage() )
 | |
|         m_book->SetSelection(1);
 | |
| 
 | |
|     // but ensure that the top of the tree is shown nevertheless
 | |
|     wxTreeCtrl * const tree = m_book->GetTreeCtrl();
 | |
| 
 | |
|     wxTreeItemIdValue cookie;
 | |
|     tree->EnsureVisible(tree->GetFirstChild(tree->GetRootItem(), cookie));
 | |
| #else
 | |
|     if ( !pageSet || !m_book->GetCurrentPage() )
 | |
|     {
 | |
|         // for other books set selection twice to force connected event handler
 | |
|         // to force lazy creation of initial visible content
 | |
|         m_book->SetSelection(1);
 | |
|         m_book->SetSelection(0);
 | |
|     }
 | |
| #endif // USE_TREEBOOK
 | |
| }
 | |
| 
 | |
| WidgetsPage *WidgetsFrame::CurrentPage()
 | |
| {
 | |
|     wxWindow *page = m_book->GetCurrentPage();
 | |
| 
 | |
| #if !USE_TREEBOOK
 | |
|     WidgetsBookCtrl *subBook = wxStaticCast(page, WidgetsBookCtrl);
 | |
|     wxCHECK_MSG( subBook, NULL, "no WidgetsBookCtrl?" );
 | |
| 
 | |
|     page = subBook->GetCurrentPage();
 | |
| #endif // !USE_TREEBOOK
 | |
| 
 | |
|     return wxStaticCast(page, WidgetsPage);
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::ConnectToWidgetEvents()
 | |
| {
 | |
|     const Widgets& widgets = CurrentPage()->GetWidgets();
 | |
| 
 | |
|     for ( Widgets::const_iterator it = widgets.begin();
 | |
|             it != widgets.end();
 | |
|             ++it )
 | |
|     {
 | |
|         wxWindow* const w = *it;
 | |
|         wxCHECK_RET(w, "NULL widget");
 | |
| 
 | |
|         w->Bind(wxEVT_SET_FOCUS, &WidgetsFrame::OnWidgetFocus, this);
 | |
|         w->Bind(wxEVT_KILL_FOCUS, &WidgetsFrame::OnWidgetFocus, this);
 | |
|     }
 | |
| }
 | |
| 
 | |
| WidgetsFrame::~WidgetsFrame()
 | |
| {
 | |
| #if USE_LOG
 | |
|     delete m_logTarget;
 | |
| #endif // USE_LOG
 | |
| }
 | |
| 
 | |
| // ----------------------------------------------------------------------------
 | |
| // WidgetsFrame event handlers
 | |
| // ----------------------------------------------------------------------------
 | |
| 
 | |
| void WidgetsFrame::OnExit(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     Close();
 | |
| }
 | |
| 
 | |
| #if USE_LOG
 | |
| void WidgetsFrame::OnButtonClearLog(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     m_lboxLog->Clear();
 | |
| }
 | |
| #endif // USE_LOG
 | |
| 
 | |
| #if wxUSE_MENUS
 | |
| 
 | |
| void WidgetsFrame::OnPageChanging(WidgetsBookCtrlEvent& event)
 | |
| {
 | |
| #if USE_TREEBOOK
 | |
|     // don't allow selection of entries without pages (categories)
 | |
|     if ( !m_book->GetPage(event.GetSelection()) )
 | |
|         event.Veto();
 | |
| #else
 | |
|     wxUnusedVar(event);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnPageChanged(WidgetsBookCtrlEvent& event)
 | |
| {
 | |
|     const int sel = event.GetSelection();
 | |
| 
 | |
|     // adjust "Page" menu selection
 | |
|     wxMenuItem *item = GetMenuBar()->FindItem(Widgets_GoToPage + sel);
 | |
|     if ( item )
 | |
|         item->Check();
 | |
| 
 | |
|     GetMenuBar()->Check(Widgets_BusyCursor, false);
 | |
| 
 | |
|     // create the pages on demand, otherwise the sample startup is too slow as
 | |
|     // it creates hundreds of controls
 | |
|     WidgetsPage *curPage = CurrentPage();
 | |
|     if ( curPage->GetChildren().empty() )
 | |
|     {
 | |
|         wxWindowUpdateLocker noUpdates(curPage);
 | |
|         curPage->CreateContent();
 | |
|         curPage->Layout();
 | |
| 
 | |
|         ConnectToWidgetEvents();
 | |
|     }
 | |
| 
 | |
|     // re-apply the attributes to the widget(s)
 | |
|     curPage->SetUpWidget();
 | |
| 
 | |
|     event.Skip();
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnGoToPage(wxCommandEvent& event)
 | |
| {
 | |
| #if USE_TREEBOOK
 | |
|     m_book->SetSelection(event.GetId() - Widgets_GoToPage);
 | |
| #else
 | |
|     m_book->SetSelection(m_book->GetPageCount()-1);
 | |
|     WidgetsBookCtrl *book = wxStaticCast(m_book->GetCurrentPage(), WidgetsBookCtrl);
 | |
|     book->SetSelection(event.GetId() - Widgets_GoToPage);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| #if wxUSE_TOOLTIPS
 | |
| 
 | |
| void WidgetsFrame::OnSetTooltip(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     wxTextEntryDialog dialog
 | |
|                       (
 | |
|                         this,
 | |
|                         "Tooltip text (may use \\n, leave empty to remove): ",
 | |
|                         "Widgets sample",
 | |
|                         WidgetsPage::GetAttrs().m_tooltip
 | |
|                       );
 | |
| 
 | |
|     if ( dialog.ShowModal() != wxID_OK )
 | |
|         return;
 | |
| 
 | |
|     WidgetsPage::GetAttrs().m_tooltip = dialog.GetValue();
 | |
|     WidgetsPage::GetAttrs().m_tooltip.Replace("\\n", "\n");
 | |
| 
 | |
|     CurrentPage()->SetUpWidget();
 | |
| }
 | |
| 
 | |
| #endif // wxUSE_TOOLTIPS
 | |
| 
 | |
| namespace
 | |
| {
 | |
| 
 | |
| // Trivial wrapper for wxGetColourFromUser() which also does something even if
 | |
| // the colour dialog is not available in the current build (which may happen
 | |
| // for the ports in development and it is still useful to see how colours work)
 | |
| wxColour GetColourFromUser(wxWindow *parent, const wxColour& colDefault)
 | |
| {
 | |
| #if wxUSE_COLOURDLG
 | |
|     return wxGetColourFromUser(parent, colDefault);
 | |
| #else // !wxUSE_COLOURDLG
 | |
|     if ( colDefault == *wxBLACK )
 | |
|         return *wxWHITE;
 | |
|     else
 | |
|         return *wxBLACK;
 | |
| #endif // wxUSE_COLOURDLG/!wxUSE_COLOURDLG
 | |
| }
 | |
| 
 | |
| } // anonymous namespace
 | |
| 
 | |
| void WidgetsFrame::OnSetFgCol(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     // allow for debugging the default colour the first time this is called
 | |
|     WidgetsPage *page = CurrentPage();
 | |
| 
 | |
|     if (!WidgetsPage::GetAttrs().m_colFg.IsOk())
 | |
|         WidgetsPage::GetAttrs().m_colFg = page->GetForegroundColour();
 | |
| 
 | |
|     wxColour col = GetColourFromUser(this, WidgetsPage::GetAttrs().m_colFg);
 | |
|     if ( !col.IsOk() )
 | |
|         return;
 | |
| 
 | |
|     WidgetsPage::GetAttrs().m_colFg = col;
 | |
| 
 | |
|     page->SetUpWidget();
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnSetBgCol(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     WidgetsPage *page = CurrentPage();
 | |
| 
 | |
|     if ( !WidgetsPage::GetAttrs().m_colBg.IsOk() )
 | |
|         WidgetsPage::GetAttrs().m_colBg = page->GetBackgroundColour();
 | |
| 
 | |
|     wxColour col = GetColourFromUser(this, WidgetsPage::GetAttrs().m_colBg);
 | |
|     if ( !col.IsOk() )
 | |
|         return;
 | |
| 
 | |
|     WidgetsPage::GetAttrs().m_colBg = col;
 | |
| 
 | |
|     page->SetUpWidget();
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnSetPageBg(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     wxColour col = GetColourFromUser(this, GetBackgroundColour());
 | |
|     if ( !col.IsOk() )
 | |
|         return;
 | |
| 
 | |
|     WidgetsPage::GetAttrs().m_colPageBg = col;
 | |
| 
 | |
|     CurrentPage()->SetUpWidget();
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnSetFont(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
| #if wxUSE_FONTDLG
 | |
|     WidgetsPage *page = CurrentPage();
 | |
| 
 | |
|     if (!WidgetsPage::GetAttrs().m_font.IsOk())
 | |
|         WidgetsPage::GetAttrs().m_font = page->GetFont();
 | |
| 
 | |
|     wxFont font = wxGetFontFromUser(this, WidgetsPage::GetAttrs().m_font);
 | |
|     if ( !font.IsOk() )
 | |
|         return;
 | |
| 
 | |
|     WidgetsPage::GetAttrs().m_font = font;
 | |
| 
 | |
|     page->SetUpWidget();
 | |
|     // The best size of the widget could have changed after changing its font,
 | |
|     // so re-layout to show it correctly.
 | |
|     page->Layout();
 | |
| #else
 | |
|     wxLogMessage("Font selection dialog not available in current build.");
 | |
| #endif
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnEnable(wxCommandEvent& event)
 | |
| {
 | |
|     WidgetsPage::GetAttrs().m_enabled = event.IsChecked();
 | |
| 
 | |
|     CurrentPage()->SetUpWidget();
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnShow(wxCommandEvent &event)
 | |
| {
 | |
|     WidgetsPage::GetAttrs().m_show = event.IsChecked();
 | |
| 
 | |
|     CurrentPage()->SetUpWidget();
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnSetBorder(wxCommandEvent& event)
 | |
| {
 | |
|     int border;
 | |
|     switch ( event.GetId() )
 | |
|     {
 | |
|         case Widgets_BorderNone:   border = wxBORDER_NONE;   break;
 | |
|         case Widgets_BorderStatic: border = wxBORDER_STATIC; break;
 | |
|         case Widgets_BorderSimple: border = wxBORDER_SIMPLE; break;
 | |
|         case Widgets_BorderRaised: border = wxBORDER_RAISED; break;
 | |
|         case Widgets_BorderSunken: border = wxBORDER_SUNKEN; break;
 | |
|         case Widgets_BorderDouble: border = wxBORDER_DOUBLE; break;
 | |
| 
 | |
|         default:
 | |
|             wxFAIL_MSG( "unknown border style" );
 | |
|             wxFALLTHROUGH;
 | |
| 
 | |
|         case Widgets_BorderDefault: border = wxBORDER_DEFAULT; break;
 | |
|     }
 | |
| 
 | |
|     WidgetsPage::GetAttrs().m_defaultFlags &= ~wxBORDER_MASK;
 | |
|     WidgetsPage::GetAttrs().m_defaultFlags |= border;
 | |
| 
 | |
|     WidgetsPage *page = CurrentPage();
 | |
| 
 | |
|     page->RecreateWidget();
 | |
| 
 | |
|     ConnectToWidgetEvents();
 | |
| 
 | |
|     // re-apply the attributes to the widget(s)
 | |
|     page->SetUpWidget();
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnSetVariant(wxCommandEvent& event)
 | |
| {
 | |
|     wxWindowVariant v;
 | |
|     switch ( event.GetId() )
 | |
|     {
 | |
|         case Widgets_VariantSmall:  v = wxWINDOW_VARIANT_SMALL; break;
 | |
|         case Widgets_VariantMini:   v = wxWINDOW_VARIANT_MINI; break;
 | |
|         case Widgets_VariantLarge:  v = wxWINDOW_VARIANT_LARGE; break;
 | |
| 
 | |
|         default:
 | |
|             wxFAIL_MSG( "unknown window variant" );
 | |
|             wxFALLTHROUGH;
 | |
| 
 | |
|         case Widgets_VariantNormal: v = wxWINDOW_VARIANT_NORMAL; break;
 | |
|     }
 | |
| 
 | |
|     WidgetsPage::GetAttrs().m_variant = v;
 | |
| 
 | |
|     CurrentPage()->SetUpWidget();
 | |
|     CurrentPage()->Layout();
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnToggleLayoutDirection(wxCommandEvent& event)
 | |
| {
 | |
|     WidgetsPage::GetAttrs().m_dir = event.IsChecked() ? wxLayout_RightToLeft
 | |
|                                        : wxLayout_LeftToRight;
 | |
| 
 | |
|     CurrentPage()->SetUpWidget();
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnToggleGlobalBusyCursor(wxCommandEvent& event)
 | |
| {
 | |
|     if ( event.IsChecked() )
 | |
|         wxBeginBusyCursor();
 | |
|     else
 | |
|         wxEndBusyCursor();
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnToggleBusyCursor(wxCommandEvent& event)
 | |
| {
 | |
|     WidgetsPage::GetAttrs().m_cursor = (event.IsChecked() ? *wxHOURGLASS_CURSOR
 | |
|                                                           : wxNullCursor);
 | |
| 
 | |
|     CurrentPage()->SetUpWidget();
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnDisableAutoComplete(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     wxTextEntryBase *entry = CurrentPage()->GetTextEntry();
 | |
|     wxCHECK_RET( entry, "menu item should be disabled" );
 | |
| 
 | |
|     if ( entry->AutoComplete(wxArrayString()) )
 | |
|     {
 | |
|         wxLogMessage("Disabled auto completion.");
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         wxLogMessage("AutoComplete() failed.");
 | |
|     }
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnAutoCompleteFixed(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     wxTextEntryBase *entry = CurrentPage()->GetTextEntry();
 | |
|     wxCHECK_RET( entry, "menu item should be disabled" );
 | |
| 
 | |
|     wxArrayString completion_choices;
 | |
| 
 | |
|     // add a few strings so a completion occurs on any letter typed
 | |
|     for ( char idxc = 'a'; idxc < 'z'; ++idxc )
 | |
|         completion_choices.push_back(wxString::Format("%c%c", idxc, idxc));
 | |
| 
 | |
|     completion_choices.push_back("is this string for test?");
 | |
|     completion_choices.push_back("this is a test string");
 | |
|     completion_choices.push_back("this is another test string");
 | |
|     completion_choices.push_back("this string is for test");
 | |
| 
 | |
|     if ( entry->AutoComplete(completion_choices) )
 | |
|     {
 | |
|         wxLogMessage("Enabled auto completion of a set of fixed strings.");
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         wxLogMessage("AutoComplete() failed.");
 | |
|     }
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnAutoCompleteFilenames(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     wxTextEntryBase *entry = CurrentPage()->GetTextEntry();
 | |
|     wxCHECK_RET( entry, "menu item should be disabled" );
 | |
| 
 | |
|     if ( entry->AutoCompleteFileNames() )
 | |
|     {
 | |
|         wxLogMessage("Enabled auto completion of file names.");
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         wxLogMessage("AutoCompleteFileNames() failed.");
 | |
|     }
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnAutoCompleteDirectories(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     wxTextEntryBase *entry = CurrentPage()->GetTextEntry();
 | |
|     wxCHECK_RET( entry, "menu item should be disabled" );
 | |
| 
 | |
|     if ( entry->AutoCompleteDirectories() )
 | |
|     {
 | |
|         wxLogMessage("Enabled auto completion of directories.");
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         wxLogMessage("AutoCompleteDirectories() failed.");
 | |
|     }
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnAutoCompleteCustom(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     DoUseCustomAutoComplete();
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::DoUseCustomAutoComplete(size_t minLength)
 | |
| {
 | |
|     wxTextEntryBase *entry = CurrentPage()->GetTextEntry();
 | |
|     wxCHECK_RET( entry, "menu item should be disabled" );
 | |
| 
 | |
|     // This is a simple (and hence rather useless) example of a custom
 | |
|     // completer class that completes the first word (only) initially and only
 | |
|     // build the list of the possible second words once the first word is
 | |
|     // known. This allows to avoid building the full 676000 item list of
 | |
|     // possible strings all at once as the we have 1000 possibilities for the
 | |
|     // first word (000..999) and 676 (aa..zz) for the second one.
 | |
|     class CustomTextCompleter : public wxTextCompleterSimple
 | |
|     {
 | |
|     public:
 | |
|         explicit CustomTextCompleter(size_t minLength)
 | |
|             : m_minLength(minLength)
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         virtual void GetCompletions(const wxString& prefix, wxArrayString& res) wxOVERRIDE
 | |
|         {
 | |
|             // This is used for illustrative purposes only and shows how many
 | |
|             // completions we return every time when we're called.
 | |
|             class LogCompletions
 | |
|             {
 | |
|             public:
 | |
|                 LogCompletions(const wxString& prefix, const wxArrayString& res)
 | |
|                     : m_prefix(prefix),
 | |
|                       m_res(res)
 | |
|                 {
 | |
|                 }
 | |
| 
 | |
|                 ~LogCompletions()
 | |
|                 {
 | |
|                     wxLogMessage("Returning %lu possible completions for "
 | |
|                                  "prefix \"%s\"",
 | |
|                                  m_res.size(), m_prefix);
 | |
|                 }
 | |
| 
 | |
|             private:
 | |
|                 const wxString& m_prefix;
 | |
|                 const wxArrayString& m_res;
 | |
|             } logCompletions(prefix, res);
 | |
| 
 | |
| 
 | |
|             // Wait for enough text to be entered before proposing completions:
 | |
|             // this is done to avoid proposing too many of them when the
 | |
|             // control is empty, for example.
 | |
|             if ( prefix.length() < m_minLength )
 | |
|                 return;
 | |
| 
 | |
|             // The only valid strings start with 3 digits so check for their
 | |
|             // presence proposing to complete the remaining ones.
 | |
|             if ( !wxIsdigit(prefix[0]) )
 | |
|                 return;
 | |
| 
 | |
|             if ( prefix.length() == 1 )
 | |
|             {
 | |
|                 for ( int i = 0; i < 10; i++ )
 | |
|                     for ( int j = 0; j < 10; j++ )
 | |
|                         res.push_back(wxString::Format("%s%02d",
 | |
|                                                        prefix, 10*i + j));
 | |
|                 return;
 | |
|             }
 | |
|             else if ( !wxIsdigit(prefix[1]) )
 | |
|                 return;
 | |
| 
 | |
|             if ( prefix.length() == 2 )
 | |
|             {
 | |
|                 for ( int i = 0; i < 10; i++ )
 | |
|                     res.push_back(wxString::Format("%s%d", prefix, i));
 | |
|                 return;
 | |
|             }
 | |
|             else if ( !wxIsdigit(prefix[2]) )
 | |
|                 return;
 | |
| 
 | |
|             // Next we must have a space and two letters.
 | |
|             wxString prefix2(prefix);
 | |
|             if ( prefix.length() == 3 )
 | |
|                 prefix2 += ' ';
 | |
|             else if ( prefix[3] != ' ' )
 | |
|                 return;
 | |
| 
 | |
|             if ( prefix2.length() == 4 )
 | |
|             {
 | |
|                 for ( char c = 'a'; c <= 'z'; c++ )
 | |
|                     for ( char d = 'a'; d <= 'z'; d++ )
 | |
|                         res.push_back(wxString::Format("%s%c%c", prefix2, c, d));
 | |
|                 return;
 | |
|             }
 | |
|             else if ( !wxIslower(prefix[4]) )
 | |
|                 return;
 | |
| 
 | |
|             if ( prefix.length() == 5 )
 | |
|             {
 | |
|                 for ( char c = 'a'; c <= 'z'; c++ )
 | |
|                     res.push_back(prefix + c);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         size_t m_minLength;
 | |
|     };
 | |
| 
 | |
|     if ( entry->AutoComplete(new CustomTextCompleter(minLength)))
 | |
|     {
 | |
|         wxLogMessage("Enabled custom auto completer for \"NNN XX\" items "
 | |
|                      "(where N is a digit and X is a letter).");
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         wxLogMessage("AutoComplete() failed.");
 | |
|     }
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnAutoCompleteKeyLength(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     const wxString message = "The auto-completion is triggered if and only if\n"
 | |
|                              "the length of the search key (prefix) is at least [LENGTH].\n"
 | |
|                              "Hint: 0 disables the length check completely.";
 | |
|     const wxString prompt = "Enter the minimum key length:";
 | |
|     const wxString caption = "Minimum key length";
 | |
| 
 | |
|     int res = wxGetNumberFromUser(message, prompt, caption, 1, 0, 100, this);
 | |
|     if ( res == -1 )
 | |
|         return;
 | |
| 
 | |
|     wxLogMessage("The minimum key length for autocomplete is %d.", res);
 | |
| 
 | |
|     DoUseCustomAutoComplete(static_cast<size_t>(res));
 | |
| }
 | |
| 
 | |
| void WidgetsFrame::OnSetHint(wxCommandEvent& WXUNUSED(event))
 | |
| {
 | |
|     wxTextEntryBase *entry = CurrentPage()->GetTextEntry();
 | |
|     wxCHECK_RET( entry, "menu item should be disabled" );
 | |
| 
 | |
|     static wxString s_hint("Type here");
 | |
|     wxString
 | |
|         hint = wxGetTextFromUser("Text hint:", "Widgets sample", s_hint, this);
 | |
|     if ( hint.empty() )
 | |
|         return;
 | |
| 
 | |
|     s_hint = hint;
 | |
| 
 | |
|     if ( entry->SetHint(hint) )
 | |
|     {
 | |
|         wxLogMessage("Set hint to \"%s\".", hint);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         wxLogMessage("Text hints not supported.");
 | |
|     }
 | |
| }
 | |
| 
 | |
| #endif // wxUSE_MENUS
 | |
| 
 | |
| void WidgetsFrame::OnWidgetFocus(wxFocusEvent& event)
 | |
| {
 | |
|     // Don't show annoying message boxes when starting or closing the sample,
 | |
|     // only log these events in our own logger.
 | |
|     if ( wxGetApp().IsUsingLogWindow() )
 | |
|     {
 | |
|         wxWindow* win = (wxWindow*)event.GetEventObject();
 | |
|         wxLogMessage("Widget '%s' %s focus", win->GetClassInfo()->GetClassName(),
 | |
|                      event.GetEventType() == wxEVT_SET_FOCUS ? "got" : "lost");
 | |
|     }
 | |
| 
 | |
|     event.Skip();
 | |
| }
 | |
| 
 | |
| // ----------------------------------------------------------------------------
 | |
| // WidgetsPageInfo
 | |
| // ----------------------------------------------------------------------------
 | |
| 
 | |
| WidgetsPageInfo::WidgetsPageInfo(Constructor ctor, const wxString& label, int categories)
 | |
|                : m_label(label)
 | |
|                , m_categories(categories)
 | |
| {
 | |
|     m_ctor = ctor;
 | |
| 
 | |
|     m_next = NULL;
 | |
| 
 | |
|     // dummy sorting: add and immediately sort in the list according to label
 | |
|     if ( WidgetsPage::ms_widgetPages )
 | |
|     {
 | |
|         WidgetsPageInfo *node_prev = WidgetsPage::ms_widgetPages;
 | |
|         if ( wxStrcmp(label, node_prev->GetLabel().c_str()) < 0 )
 | |
|         {
 | |
|             // add as first
 | |
|             m_next = node_prev;
 | |
|             WidgetsPage::ms_widgetPages = this;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             WidgetsPageInfo *node_next;
 | |
|             do
 | |
|             {
 | |
|                 node_next = node_prev->GetNext();
 | |
|                 if ( node_next )
 | |
|                 {
 | |
|                     // add if between two
 | |
|                     if ( wxStrcmp(label, node_next->GetLabel().c_str()) < 0 )
 | |
|                     {
 | |
|                         node_prev->SetNext(this);
 | |
|                         m_next = node_next;
 | |
|                         // force to break loop
 | |
|                         node_next = NULL;
 | |
|                     }
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     // add as last
 | |
|                     node_prev->SetNext(this);
 | |
|                     m_next = node_next;
 | |
|                 }
 | |
|                 node_prev = node_next;
 | |
|             }
 | |
|             while ( node_next );
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         // add when first
 | |
|         WidgetsPage::ms_widgetPages = this;
 | |
|     }
 | |
| }
 | |
| 
 | |
| // ----------------------------------------------------------------------------
 | |
| // WidgetsPage
 | |
| // ----------------------------------------------------------------------------
 | |
| 
 | |
| WidgetsPageInfo *WidgetsPage::ms_widgetPages = NULL;
 | |
| 
 | |
| WidgetsPage::WidgetsPage(WidgetsBookCtrl *book,
 | |
|                          wxImageList *imaglist,
 | |
|                          const char *const icon[])
 | |
|            : wxPanel(book, wxID_ANY,
 | |
|                      wxDefaultPosition, wxDefaultSize,
 | |
|                      wxCLIP_CHILDREN |
 | |
|                      wxTAB_TRAVERSAL)
 | |
| {
 | |
|     imaglist->Add(wxBitmap(wxImage(icon).Scale(ICON_SIZE, ICON_SIZE)));
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| WidgetAttributes& WidgetsPage::GetAttrs()
 | |
| {
 | |
|     static WidgetAttributes s_attrs;
 | |
| 
 | |
|     return s_attrs;
 | |
| }
 | |
| 
 | |
| void WidgetsPage::SetUpWidget()
 | |
| {
 | |
|     const Widgets widgets = GetWidgets();
 | |
| 
 | |
|     for ( Widgets::const_iterator it = widgets.begin();
 | |
|             it != widgets.end();
 | |
|             ++it )
 | |
|     {
 | |
|         wxCHECK_RET(*it, "NULL widget");
 | |
| 
 | |
| #if wxUSE_TOOLTIPS
 | |
|         (*it)->SetToolTip(GetAttrs().m_tooltip);
 | |
| #endif // wxUSE_TOOLTIPS
 | |
| #if wxUSE_FONTDLG
 | |
|         if ( GetAttrs().m_font.IsOk() )
 | |
|         {
 | |
|             (*it)->SetFont(GetAttrs().m_font);
 | |
|         }
 | |
| #endif // wxUSE_FONTDLG
 | |
|         if ( GetAttrs().m_colFg.IsOk() )
 | |
|         {
 | |
|             (*it)->SetForegroundColour(GetAttrs().m_colFg);
 | |
|         }
 | |
| 
 | |
|         if ( GetAttrs().m_colBg.IsOk() )
 | |
|         {
 | |
|             (*it)->SetBackgroundColour(GetAttrs().m_colBg);
 | |
|         }
 | |
| 
 | |
|         (*it)->SetLayoutDirection(GetAttrs().m_dir);
 | |
|         (*it)->Enable(GetAttrs().m_enabled);
 | |
|         (*it)->Show(GetAttrs().m_show);
 | |
| 
 | |
|         (*it)->SetCursor(GetAttrs().m_cursor);
 | |
| 
 | |
|         (*it)->SetWindowVariant(GetAttrs().m_variant);
 | |
| 
 | |
|         (*it)->Refresh();
 | |
|     }
 | |
| 
 | |
|     if ( GetAttrs().m_colPageBg.IsOk() )
 | |
|     {
 | |
|         SetBackgroundColour(GetAttrs().m_colPageBg);
 | |
|         Refresh();
 | |
|     }
 | |
| }
 | |
| 
 | |
| wxSizer *WidgetsPage::CreateSizerWithText(wxControl *control,
 | |
|                                           wxWindowID id,
 | |
|                                           wxTextCtrl **ppText)
 | |
| {
 | |
|     wxSizer *sizerRow = new wxBoxSizer(wxHORIZONTAL);
 | |
|     wxTextCtrl *text = new wxTextCtrl(this, id, wxEmptyString,
 | |
|         wxDefaultPosition, wxDefaultSize, wxTE_PROCESS_ENTER);
 | |
| 
 | |
|     sizerRow->Add(control, wxSizerFlags(0).Border(wxRIGHT).CentreVertical());
 | |
|     sizerRow->Add(text, wxSizerFlags(1).Border(wxLEFT).CentreVertical());
 | |
| 
 | |
|     if ( ppText )
 | |
|         *ppText = text;
 | |
| 
 | |
|     return sizerRow;
 | |
| }
 | |
| 
 | |
| // create a sizer containing a label and a text ctrl
 | |
| wxSizer *WidgetsPage::CreateSizerWithTextAndLabel(const wxString& label,
 | |
|                                                   wxWindowID id,
 | |
|                                                   wxTextCtrl **ppText)
 | |
| {
 | |
|     return CreateSizerWithText(new wxStaticText(this, wxID_ANY, label),
 | |
|         id, ppText);
 | |
| }
 | |
| 
 | |
| // create a sizer containing a button and a text ctrl
 | |
| wxSizer *WidgetsPage::CreateSizerWithTextAndButton(wxWindowID idBtn,
 | |
|                                                    const wxString& label,
 | |
|                                                    wxWindowID id,
 | |
|                                                    wxTextCtrl **ppText)
 | |
| {
 | |
|     return CreateSizerWithText(new wxButton(this, idBtn, label), id, ppText);
 | |
| }
 | |
| 
 | |
| wxCheckBox *WidgetsPage::CreateCheckBoxAndAddToSizer(wxSizer *sizer,
 | |
|                                                      const wxString& label,
 | |
|                                                      wxWindowID id)
 | |
| {
 | |
|     wxCheckBox *checkbox = new wxCheckBox(this, id, label);
 | |
|     sizer->Add(checkbox, wxSizerFlags().HorzBorder());
 | |
|     sizer->AddSpacer(2);
 | |
| 
 | |
|     return checkbox;
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| bool WidgetsPage::IsUsingLogWindow()
 | |
| {
 | |
|     return wxGetApp().IsUsingLogWindow();
 | |
| }
 |