[ZETCODE]wxWidgets教程九:组件专题2
本教程原文链接:http://zetcode.com/gui/wxwidgets/widgetsII/
翻译:瓶哥
日期:2013年12月15日星期日
主页:http://www.cnblogs.com/pingge/
若有翻译错误或者歧义请联系我!
在这一章中,我们会继续介绍一些组件的知识,我们会提到wxListBox,wxNotebool和wxScrolledWindow。
wxListBox
一个wxListBox组件被用来显示一些元素,它是一个有一系列字符串的矩形。我们可以使用它来显示一个MP3文件的列表,一些书名,或者是一个大工程的模块名。一个wxListBox可以以两种不同的状态被创建。一个单一选择的状态或者一个多重选择的状态。单一状态是默认的状态,在wxListBox里面有两个事件类型,第一个是wxEVT_COMMAND_LISTBOX_SELECTED,当我们选择wxListBox里面的一个元素时这个事件产生;第二个是wxEVT_COMMAND_LISTBOX_DOUBLE_CLICKED,当我们双击wxListBox里面的一个元素时这个事件产生。在wxListBox内部的元素在GTK的平台上有限制,大概不能超过2000个左右的元素。元素的ID从0开始计算,滚动条会在需要的时候自动生成。
listbox.h
#include <wx/wx.h>
#include <wx/listbox.h> class MyPanel : public wxPanel
{
public:
MyPanel(wxPanel * parent); void OnNew(wxCommandEvent & event);
void OnRename(wxCommandEvent & event);
void OnClear(wxCommandEvent & event);
void OnDelete(wxCommandEvent & event); wxListBox * m_lb; wxButton * m_newb;
wxButton * m_renameb;
wxButton * m_clearb;
wxButton * m_deleteb;
}; class Listbox : public wxFrame
{
public:
Listbox(const wxString & title); void OnDbClick(wxCommandEvent & event); wxListBox * listbox;
MyPanel * btnPanel;
}; enum{ID_RENAME, ID_LISTBOX};
listbox.cpp
#include "listbox.h"
#include <wx/textdlg.h> Listbox::Listbox(const wxString & title)
: wxFrame(NULL, wxID_ANY, title, wxDefaultPosition, wxSize(270, 200))
{
wxPanel * panel = new wxPanel(this, wxID_ANY); wxBoxSizer * hbox = new wxBoxSizer(wxHORIZONTAL); listbox = new wxListBox(panel, ID_LISTBOX, wxDefaultPosition, wxDefaultSize); btnPanel = new MyPanel(panel); hbox->Add(listbox, 3, wxEXPAND | wxALL, 20); hbox->Add(btnPanel, 2, wxEXPAND | wxRIGHT, 10); Connect(wxEVT_COMMAND_LISTBOX_DOUBLECLICKED,
wxCommandEventHandler(Listbox::OnDbClick)); panel->SetSizer(hbox); Centre();
} MyPanel::MyPanel(wxPanel * parent)
: wxPanel(parent, wxID_ANY)
{
wxBoxSizer * vbox = new wxBoxSizer(wxVERTICAL); // GetParent()方法获得了panel的父组件的地址
Listbox * lb = (Listbox*)parent->GetParent(); m_lb = lb->listbox; m_newb = new wxButton(this, wxID_NEW, _T("New"));
m_renameb = new wxButton(this, ID_RENAME, _T("Rename"));
m_deleteb = new wxButton(this, wxID_DELETE, _T("Delete"));
m_clearb = new wxButton(this, wxID_CLEAR, _T("Clear")); Connect(wxID_NEW, wxEVT_COMMAND_BUTTON_CLICKED,
wxCommandEventHandler(MyPanel::OnNew) );
Connect(ID_RENAME, wxEVT_COMMAND_BUTTON_CLICKED,
wxCommandEventHandler(MyPanel::OnRename) );
Connect(wxID_CLEAR, wxEVT_COMMAND_BUTTON_CLICKED,
wxCommandEventHandler(MyPanel::OnClear) );
Connect(wxID_DELETE, wxEVT_COMMAND_BUTTON_CLICKED,
wxCommandEventHandler(MyPanel::OnDelete) ); vbox->Add(-1, 20);
vbox->Add(m_newb);
vbox->Add(m_renameb, 0, wxTOP, 5);
vbox->Add(m_deleteb, 0, wxTOP, 5);
vbox->Add(m_clearb, 0, wxTOP, 5); SetSizer(vbox);
} void MyPanel::OnNew(wxCommandEvent & event)
{
wxString str = wxGetTextFromUser(_T("Add new item"));
if(str.Len() > 0)
{
m_lb->Append(str);
}
} void MyPanel::OnClear(wxCommandEvent & event)
{
m_lb->Clear();
} void MyPanel::OnRename(wxCommandEvent & event)
{
wxString text;
wxString renamed; int sel = m_lb->GetSelection();
if(sel != -1)
{
text = m_lb->GetString(sel);
renamed = wxGetTextFromUser(_T("Rename item"), _T("Rename dialog"), text);
}
if(!renamed.IsEmpty())
{
m_lb->Delete(sel);
m_lb->Insert(renamed, sel);
}
} void MyPanel::OnDelete(wxCommandEvent & event)
{
int sel = m_lb->GetSelection();
if(sel != -1)
{
m_lb->Delete(sel);
}
} void Listbox::OnDbClick(wxCommandEvent & event)
{
wxString text;
wxString renamed; int sel = listbox->GetSelection();
if(sel != -1)
{
text = listbox->GetString(sel);
renamed = wxGetTextFromUser(_T("Rename item"), _T("Rename dialog"), text);
}
if(!renamed.IsEmpty())
{
listbox->Delete(sel);
listbox->Insert(renamed, sel);
}
}
main.h
#include <wx/wx.h> class MyApp : public wxApp
{
public:
virtual bool OnInit();
};
main.cpp
#include "main.h"
#include "Listbox.h" IMPLEMENT_APP(MyApp) bool MyApp::OnInit()
{
Listbox * listbox = new Listbox(_T("Listbox"));
listbox->Show(true); return true;
}
listbox = new wxListBox(panel, ID_LISTBOX, wxDEFAULT_POSITION, wxDEFAULT_SIZE);
这个是listbox组件的构造器.
在我们的例子中,我们有一列四个按钮,这些按钮用来在listbox内添加、修改、删除、清空元素。
wxString str = wxGetTextFromUser(_T("Add new item"));
if(str.Len() > 0)
{m_lb->Append(str);
}
我们使用上面的代码显示一个wxGetTextFromUser对话框接收用户输入,然后调用Append()函数添加到listbox内。
m_lb->Clear();
清空listbox只需要简单的调用一个Clear()方法。
int sel = m_lb->GetSelection();
if(sel != -1)
{m_lb->Delete(sel);
}
我们调用GetSelection()方法计算出当前选中的是第几项,然后我们调用Delete()方法删除元素。
修改一个元素需要以下几个步骤:
wxString text;
wxString renamed;
我们定义了两个字符串对象。
int sel = listbox->GetSelection();
if(sel != -1)
{text = listbox->GetString(sel);
Renamed = wxGetTextFromUser(_T("Rename item"), _T("Rename dialog"), text);
}
我们获取了当前选中的的字符串,并保存起来,然后请求用户的输入。
if(!renamed.IsEmpty())
{m_lb->Delete(sel);
m_lb->Insert(renamed, sel);
}
我们检查了renamed字符串是否为空,避免了插入空字符串,然后我们删除了旧的元素,插入新的元素。

wxNotebook
wxNotebook可以在一个窗体上通过标签列举多个窗口。可以通过下面四个标识符指定标签的位置:
- wxNB_LEFT
- wxNB_RIGHT
- wxNB_TOP
- wxNB_BOTTOM
默认的是wxNB_TOP
notebook.h
#include <wx/wx.h>
#include <wx/notebook.h>
#include <wx/grid.h> class Notebook : public wxFrame
{
public:
Notebook(const wxString & title); void OnQuit(wxCommandEvent & event);
}; class MyGrid : public wxGrid
{
public:
MyGrid(wxNotebook * parent);
};
notebook.cpp
#include "notebook.h" Notebook::Notebook(const wxString & title)
: wxFrame(NULL, wxID_ANY, title, wxDefaultPosition, wxSize(400, 350))
{
wxNotebook * nb = new wxNotebook(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxNB_BOTTOM); wxMenuBar * menubar = new wxMenuBar();
wxMenu * file = new wxMenu();
file->Append(wxID_EXIT, _T("Quit"), _T(""));
menubar->Append(file, _T("&File"));
SetMenuBar(menubar); Connect(wxEVT_COMMAND_MENU_SELECTED,
wxCommandEventHandler(Notebook::OnQuit));
MyGrid * grid1 = new MyGrid(nb);
MyGrid * grid2 = new MyGrid(nb);
MyGrid * grid3 = new MyGrid(nb); nb->AddPage(grid1, _T("Sheet1"));
nb->AddPage(grid2, _T("Sheet2"));
nb->AddPage(grid3, _T("Sheet3")); CreateStatusBar();
Centre();
} void Notebook::OnQuit(wxCommandEvent & event)
{
Close(true);
} MyGrid::MyGrid(wxNotebook * parent)
: wxGrid(parent, wxID_ANY)
{
CreateGrid(30, 30);
SetRowLabelSize(50);
SetColLabelSize(25);
SetRowLabelAlignment(wxALIGN_RIGHT, wxALIGN_CENTER);
SetLabelFont(wxFont(9, wxFONTFAMILY_DEFAULT, wxFONTSTYLE_NORMAL, wxFONTWEIGHT_BOLD)); for(int i = 0; i < 30; i++)
{
this->SetRowSize(i, 25);
}
}
main.h
#include <wx/wx.h> class MyApp : public wxApp
{
public:
virtual bool OnInit();
};
main.cpp
#include "main.h"
#include "notebook.h" IMPLEMENT_APP(MyApp) bool MyApp::OnInit()
{
Notebook * notebook = new Notebook(_T("Notebook"));
notebook->Show(true); return true;
}
在这个例子中,我们创建了一个包含三个表格控件的notebook组件,notebook的标签被放置在窗体底部。
wxNotebook * nb = new wxNotebook(this, wxID_ANY, wxDEFAULTPOSITION, wxDEFAULTSIZE, wxNB_BOTTOM);
这里我们创建了一个notebook组件。
nb->AddPage(grid1, _T("Sheet1"));
nb->AddPage(grid2, _T("Sheet2"));
nb->AddPage(grid3, _T("Sheet3"));
我们添加了三个表格控件到notebook组件中。

wxScrolledWindow
这是容器组件中的其中一个。当我们有一个比屏幕还要大的区域需要显示时,这个组件很有用。在我们的例子中,我们示范了它的用法,我们在窗口中放置了一个巨大的图片。当窗口比我们的图片小的时候,滚动条会自动显示。
scrolledwindow.h
#include <wx/wx.h> class ScrolledWindow : public wxFrame
{
public:
ScrolledWindow(const wxString & title);
};
scrolledwindow.cpp
#include "scrolledwindow.h" ScrolledWindow::ScrolledWindow(const wxString & title)
: wxFrame(NULL, wxID_ANY, title, wxDefaultPosition, wxSize(300, 200))
{
wxImage::AddHandler(new wxJPEGHandler); wxScrolledWindow * sw = new wxScrolledWindow(this); wxBitmap bmp(_T("test.jpg"), wxBITMAP_TYPE_JPEG);
wxStaticBitmap * sb = new wxStaticBitmap(sw, -1, bmp); int width = bmp.GetWidth();
int height = bmp.GetHeight(); sw->SetScrollbars(10, 10, width / 10, height / 10);
sw->Scroll(50, 10); Centre();
}
main.h
#include <wx/wx.h> class MyApp : public wxApp
{
public:
virtual bool OnInit();
};
main.cpp
#include "main.h"
#include "scrolledwindow.h" IMPLEMENT_APP(MyApp) bool MyApp::OnInit()
{
ScrolledWindow * sw = new ScrolledWindow(_T("ScrolledWindow"));
sw->Show(true); return true;
}
在我们的例子中,我们显示了一张海贼王的图片。
wxImage::AddHandler(new wxJPEGHandler);
我们必须初始化wxJPEGHandler,才能处理jpg图像。
wxScrolledWindow * sw = new wxScrolledWindow(this);
wxBitmap bmp(_T("test.jpg"), wxBITMAP_TYPE_JPEG);
wxStaticBitmap * sb = new wxStaticBitmap(sw, -1, bmp);
我们创建了一个滚动窗口,并放置了一张静态图。
sw->SetScrollbars(10, 10, width / 10, height / 10);
我们设置了滚动条。
sw->Scroll(50, 10);
滚动到初始位置。

在这一章中,我们继续介绍了wxWidgets库中的组件。
[ZETCODE]wxWidgets教程九:组件专题2的更多相关文章
- [ZETCODE]wxWidgets教程八:组件专题1
本教程原文链接:http://zetcode.com/gui/wxwidgets/widgets/ 翻译:瓶哥 日期:2013年12月12日星期四 邮箱:414236069@qq.com 主页:htt ...
- [ZETCODE]wxWidgets教程七:对话框
本教程原文链接:http://zetcode.com/gui/wxwidgets/dialogs/ 翻译:瓶哥 日期:2013年12月9日星期一 邮箱:414236069@qq.com 主页:http ...
- [ZETCODE]wxWidgets教程五:布局管理
本教程原文链接:http://zetcode.com/gui/wxwidgets/layoutmanagement/ 翻译:瓶哥 日期:2013年12月4日星期三 邮箱:414236069@qq.co ...
- [ZETCODE]wxWidgets教程四:菜单栏和工具栏
本教程原文链接:http://zetcode.com/gui/wxwidgets/menustoolbars/ 翻译:瓶哥 日期:2013年11月28日星期四 邮箱:414236069@qq.com ...
- [ZETCODE]wxWidgets教程三:第一个窗体程序
本教程原文链接:http://zetcode.com/gui/wxwidgets/firstprograms/ 翻译:瓶哥 日期:2013年11月27日星期三 邮箱:414236069@qq.com ...
- [ZETCODE]wxWidgets教程六:事件处理
本教程原文链接:http://zetcode.com/gui/wxwidgets/events/ 翻译:瓶哥 日期:2013年12月7号星期六 邮箱:414236069@qq.com 主页:http: ...
- [ZETCODE]wxWidgets教程二:辅助类
本教程原文链接:http://zetcode.com/gui/wxwidgets/helperclasses/ 翻译:瓶哥 日期:2013年11月27日星期三 邮箱:414236069@qq.com ...
- [ZETCODE]wxWidgets教程一:介紹
本教程原文链接:http://zetcode.com/gui/wxwidgets/introduction/ 翻译:瓶哥 日期:2013年11月26日星期二 邮箱: 414236069@qq.com ...
- CRL快速开发框架系列教程九(导入/导出数据)
本系列目录 CRL快速开发框架系列教程一(Code First数据表不需再关心) CRL快速开发框架系列教程二(基于Lambda表达式查询) CRL快速开发框架系列教程三(更新数据) CRL快速开发框 ...
随机推荐
- 九度OJ 1042 Coincidence -- 动态规划(最长公共子序列)
题目地址:http://ac.jobdu.com/problem.php?pid=1042 题目描述: Find a longest common subsequence of two strings ...
- 24种设计模式--责任链模式【Chain ofResponsibility Pattern】
中国古代对妇女制定了“三从四德”的道德规范,“三从”是指“未嫁从父.既嫁从夫.夫死从子”,也就是说一个女性,在没有结婚的时候要听从于父亲,结了婚后听从于丈夫,丈夫死了还要听儿子的,举个例子来说,一个女 ...
- thinkphp 整合 ucenter
http://xcodebox.com/2013/06/8855.html 参考 1 ucenter源码目录下 /advanced/examples/api目录 copy到thinkphp项目根目 ...
- ASP.NET MVC5 easyui 之 treegrid 初用记录
菜鸟初次使用,参考论坛中介绍的方法仍走了一些弯路,把自己遇到的问题记录下来. 1.必须定义根节点: 2.根节点一个或多个均可: 4.根节点的父节点属性不必定义,或者定义为0: 5.各级子节点的父节点属 ...
- svn-主副分支使用
主改bug 副加功能, :主合并到副(在副中切换主分支),副调试成功,合并回主(在主切换回副分支) 奇葩的实现了需求 主改bug 副加功能, :主合并到副(在副中切换主分支),副调试成功,合并回主(在 ...
- slivelight5和数据库交互
最近开始研究sliverlight和数据库交互了,无奈网上资料较少,查阅了大量资料终于成功了,但是我记得还有别的方法,希望大家讨论一下 数据访问层我的用的是ado.net实体数据模型 然后新建了一个w ...
- python的特殊方法:
__str__和__repr__ 如果要把一个类的实例变成 str,就需要实现特殊方法__str__(): class Person(object): def __init__(self, name, ...
- pythom 安装MySQL-pythom的问题
链接一:http://blog.csdn.net/dqatsh/article/details/2418663 链接二:http://codingnow.cn/language/159.html 链接 ...
- Tmux:终端复用器
转自Tmux:终端复用器 Tmux 是一个 C 语言编写的终端,它能够在单一窗口中同时访问和控制多个终端.它是一个类似于GNU Screen 的工具.使用它,用户可以在 Linux 系统上管理多个任务 ...
- DJANGO,获取当前用户名,用户组名,用户组权限
样例,为下一步自定义用户权限作一下代码准备: def get_context_data(self, **kwargs): if self.request.user.is_authenticated() ...