3d9656395a
Add IsUsingLogWindow() that can be used to check if the log messages go into the listbox instead of being shown in a message box, which was annoying when starting or quitting the sample. This is a bit more complicated than the hack previously used in TextWidgetsPage::OnText(), but more general and can be easily reused for the focus loss messages, for example.
1430 lines
42 KiB
C++
1430 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"
|
|
|
|
#ifdef __BORLANDC__
|
|
#pragma hdrstop
|
|
#endif
|
|
|
|
// 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->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
|
|
: wxSTANDARD_CURSOR);
|
|
|
|
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() )
|
|
{
|
|
wxLogMessage("Widgets %s focus",
|
|
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,
|
|
wxNO_FULL_REPAINT_ON_RESIZE |
|
|
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);
|
|
|
|
if ( GetAttrs().m_cursor.IsOk() )
|
|
{
|
|
(*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();
|
|
}
|