在MS Windows
中,一个窗口可以分割成若干个子窗口,每一个子窗口称作一个窗片(pane),每个窗片可以独立控制,这给界面设计提供了很大的方便。
利用VC
可以很方便地实现分割窗口。分割的方法有两种:动态和静态。动态分割时可以根据用户的需要分割成数目不同的窗片,但所有窗片的属性和父窗口都是一样的;而静态分割的窗片的数目在程序中指定,运行时是固定的,但每个窗片可以有各自不同类型的视(View),因此其使用范围更为广泛。本文所讨论的问题仅限于静态分割。
窗片中视的类型大多是在主窗口的创建过程中指定的。这也就意味着,一个窗片虽然可以显示任意类型的视,但是这种类型一旦确定,在程序运行过程中就难以改变。
一、我要的是这样的!
但是我们有时确实需要改变一个窗片所显示的视的类型,也就是说,需要让一个窗片显示多种类型的视。例如一个窗口被分割成两部分,一边是命令窗口,另一边是工作窗口,根据命令窗口中发出的不同命令,需要变换不同的工作类型,这就需要工作窗口中能够显示多种类型的视窗,那么,如何做到这一点呢?
二、你可以这样做!
从图1 中可以看到,本程序共有三个视类,分别是:
* 命令视类CCmdView:用来控制右边窗片中不同视的显示;
* 选项按钮视类CRdiView:显示在右窗片中的选项视类;
* 检查按钮视类CChkView:显示在右窗片中的检查视类。
这三个视类都是CFormView 的子类。
下面我们来看如何在右窗片内进行两类视间的切换。实际上,由视A
切换到视B 的原理很简单,那就是:
1. 从窗片中删除视A;
2. 往窗片中添加视B。
步骤1 的实现非常简单,仅用一条语句即可:
m_wndSplitter.DeleteView(0, 1);
但它是必不可少的,因为你不能让一个窗片同时包含两个视。我本来希望往一个窗片中添加新的视时,VC
会自动将原来的视删掉,可是它不干。
我们来看如何实现步骤2,当一个窗片是空的时候,怎样往里面添加一个视呢?其实这样的功能在程序里我们已经用过了,看下面的语句:
BOOL CMainFrame::OnCreateClient
(LPCREATESTRUCT lpcs, CCreateContext* pContext)
{
......
if (!m_wndSplitter.CreateView(0, 0,
pContext->m_pNewViewClass,
size,
pContext))
......
}
是的,用的就是CSplitterWnd::CreateView(),要注意的是它共有五个参数,其中前两个用来指定分割窗口的窗片,第三个用来指定视的类型,第四个指定视的大小。最后的一个我们暂时用不上,用空值NULL
就可以了。
这样我们就可以编写视切换的代码了。因为视切换要操纵m_wndSplitter,而它是主窗口的成员,因此切换过程最好设计为主窗口的成员函数。但是切换命令是CCmdView
接受的,因而可以让CCmdView
接受到视更改消息后,将消息传给主窗口,由主窗口完成视更改。具体的代码是这样的:
命令视类中的消息映射:
BEGIN_MESSAGE_MAP(CCmdView, CFormView)
......
ON_BN_CLICKED(IDC_CHECK, OnSwitchToCheckView)
ON_BN_CLICKED(IDC_RADIO, OnSwitchToRadioView)
......
END_MESSAGE_MAP()
命令视类中的消息响应:
void CCmdView::OnSwitchToCheckView()
{
AfxGetApp()->m_pMainWnd->
SendMessage(WM_COMMAND, ID_CHECK);
}
void CCmdView::OnSwitchToRadioView()
{
AfxGetApp()->m_pMainWnd->
SendMessage(WM_COMMAND, ID_RADIO);
}
主窗口中的消息映射:
BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd)
......
ON_COMMAND(ID_CHECK, OnSwitchToCheckView)
ON_COMMAND(ID_RADIO, OnSwitchToRadioView)
......
END_MESSAGE_MAP()
主窗口中的消息响应:
void CMainFrame::OnSwitchToCheckView()
{
m_wndSplitter.DeleteView(0, 1);
m_wndSplitter.CreateView(0, 1,
RUNTIME_CLASS(CChkView),
CSize(0, 0),
NULL);
m_wndSplitter.RecalcLayout();
}
void CMainFrame::OnSwitchToRadioView()
{
m_wndSplitter.DeleteView(0, 1);
m_wndSplitter.CreateView(0, 1,
RUNTIME_CLASS(CRdiView),
CSize(0, 0),
NULL);
m_wndSplitter.RecalcLayout();
}
好啦,运行一下这个程序,感觉是否不错?看来大功告成了,可是……
三、还有一个问题
在运行我们辛辛苦苦编出来的程序时,回头看看VC
的调试窗口,你会发现有很多行这样的话:
Create view without document.
这是说我们创建了视,可是没有相应的文档。好在这只是警告信息,不是什么错误,如果你不需要相应的文档,就完全不用去管它。可是,VC
中一种很重要的结构就是文档-
视结构,利用这种结构,对数据操纵起来非常方便。如果需要建立与视相对应的文档,应该怎么办呢?
这就涉及到VC 中文档-
视结构的知识,不过不用怕麻烦,与本文有关的就只有这么两点而已:
1. 利用VC 创建的应用程序一般都会管理一些文档模板(Document
Template),文档类和视类的对应关系就是在文档模板里描述的。
2.
一个文档可以有多个视,创建视的时候,需要根据文档和视的对应关系,给出它所依附的文档。
怎样实现上述第一点呢?
首先建立相应的文档类:CRdiDoc 和CChkDoc。
其次是定义相应的文档模板,这是应用类的成员变量。因为在别的类中要使用它们,我们将之定义为公共类型:
class CViewSwitcherApp : public CWinApp
{
......
public:
CSingleDocTemplate* m_pRdiDocTemplate;
CSingleDocTemplate* m_pChkDocTemplate;
......
}
然后创建这两个文档模板,并加入到模板列表中:
BOOL CViewSwitcherApp::InitInstance()
{
......
m_pRdiDocTemplate = new CSingleDocTemplate(
IDR_MAINFRAME,
RUNTIME_CLASS(CRdiDoc),
RUNTIME_CLASS(CMainFrame),
RUNTIME_CLASS(CRdiView));
AddDocTemplate(m_pRdiDocTemplate);
m_pChkDocTemplate = new CSingleDocTemplate(
IDR_MAINFRAME,
RUNTIME_CLASS(CChkDoc),
RUNTIME_CLASS(CMainFrame),
RUNTIME_CLASS(CChkView));
AddDocTemplate(m_pChkDocTemplate);
......
}
至于第二点,是在创建视时完成的。还记得创建视的情况么?当时有一个叫做pCreateContext
的参数,我们将之置为空,这里就要用到它了。
pCreateContext 是一个指向被称作" 创建上下文"(CreateContext)
结构的指针,这个结构中保存一些与创建视相关的内容。在创建主窗口时,系统会构造这样一个结构,并将它作为参数传递到与创建视有关的函数中。但现在我们不创建主窗口,因此不得不自己构造这样一个结构。实际上,该结构中我们所要使用的字段只有三个:
1. 新视所属的文档模板m_pNewDocTemplate;
2. 新视的类型m_pNewViewClass;
3. 新视所属的文档m_pCurrentDoc;
其中仅有第三项需要新建,前两项都是已知的,只要指定即可。以切换到选项视为例,修改后的代码是:
void CMainFrame::OnSwitchToRadioView()
{
m_wndSplitter.DeleteView(0, 1);
CCreateContext createContext;
// 定义并初始化CreateContext
// 获取新视所属的文档模板
CSingleDocTemplate* pDocTemplate =
((CViewSwitcherApp*)AfxGetApp())-> m_pRdiDocTemplate;
// 创建新文档并初始化
CDocument* pDoc = pDocTemplate->CreateNewDocument();
pDoc->OnNewDocument();
// 设置CreateContext 相关字段
createContext.m_pNewViewClass = RUNTIME_CLASS(CChkView);
createContext.m_pCurrentDoc = pDoc;
createContext.m_pNewDocTemplate = pDocTemplate;
m_wndSplitter.CreateView(0, 1,
RUNTIME_CLASS(CRdiView),
CSize(0, 0),
&createContext);
m_wndSplitter.RecalcLayout();
}
四、最后的修改
为了使这个程序更符合要求,我们还要做一些与更换视无关的修改。在这个程序中我们一共定义了三种类型的文档,程序启动时一般要新建一个文档开始工作,可是它不知道要选择哪一种,就弹出一个对话框来询问。而这是我们不希望看到的。修改的方法是不让VC
选择新文档类型,而我们指定创建哪一种类型的文档,即把CViewSwitcherApp::CViewSwitcherApp()
中的语句
if (!ProcessShellCommand(cmdInfo)) return FALSE;
更改为
m_pDocTemplate->OpenDocumentFile(NULL)。