• 使用CTabView实现多视图(一体多面)


    一般书中介绍的是使用CSplitterWnd来拆分窗口实现多视图,CSplitterWnd中的CreateClient可以保存其创建的pCreateContext指针,以便子视图共享Document。这我用一篇文章详细说明。CTabView建立多视图的好处在于简单的标签切换,可以让每个View的窗口都很大。下面说说如何操作:

    在建立项目时,程序向导最后一步并未提供让View为CTabView类型,因此我们需要自己新建一个CTabView的子类。

    在vc2008中,在项目菜单中“项目”->“添加类”,可以自动生成类,像vc6的话ClassWizard也可以自动添加。

    修改WinApp中的InitInstance中的new CMultiDocTemplate(),修改其中的View类型为刚刚新建的CTabView子类。

        // 注册应用程序的文档模板。文档模板
    // 将用作文档、框架窗口和视图之间的连接
    CMultiDocTemplate* pDocTemplate;
    pDocTemplate = new CMultiDocTemplate(IDR_TestTabViewTYPE,
    RUNTIME_CLASS(CTestTabViewDoc),
    RUNTIME_CLASS(CChildFrame), // 自定义 MDI 子框架
    RUNTIME_CLASS(CMyTabView));
    if (!pDocTemplate)
    return FALSE;
    AddDocTemplate(pDocTemplate);

    修改完成后,可以看出TabView的形态已经出来了,不过里面没有子View。下面来添加子view。

    一般子view也需要新建,继续使用自动新建类,可以建立CFormView、ClistView等等的视图来显示数据。

    新建子view在CTabView的OnCreate函数中,在新建类中需要新建CCreateContext来将Doc给新的View。

    int CMyTabView::OnCreate(LPCREATESTRUCT lpCreateStruct)
    {
    if (CTabView::OnCreate(lpCreateStruct) == -1)
    return -1;
    // TODO: 在此添加您专用的创建代码
    CCreateContext context;
    context.m_pNewViewClass = RUNTIME_CLASS(CView1);
    context.m_pCurrentDoc = GetDocument();

    AddView(RUNTIME_CLASS(CView1),"哈哈",100,&context);
    context.m_pNewViewClass = RUNTIME_CLASS(CEditView);
    AddView(RUNTIME_CLASS(CEditView),"呵呵",101,&context);

    return 0;
    }

    上述代码新建了2个视图,CCreateContext指定需要新建View,指定Doc为TabView的Doc,在AddView中的最后参数中将CCreateContext传入。

    至此,子视图中可以使用GetDocument来得到在WinApp中CTestTabViewDoc的类实例指针,这些子视图共享了同一份Doc。

    后续:

    过了几天,我无意间发现AddView的代码,是自动将自己的Doc传给子tab,请看代码:

    int CTabView::AddView(CRuntimeClass* pViewClass, const CString& strViewLabel, int iIndex /*= -1*/, CCreateContext* pContext/* = NULL*/)
    {
    ASSERT_VALID(this);
    ENSURE(pViewClass != NULL);
    ENSURE(pViewClass->IsDerivedFrom(RUNTIME_CLASS(CView)));

    CView* pView = DYNAMIC_DOWNCAST(CView, pViewClass->CreateObject());
    ASSERT_VALID(pView);

    if (!pView->Create(NULL, _T(""), WS_CHILD | WS_VISIBLE, CRect(0, 0, 0, 0), &m_wndTabs, (UINT) -1, pContext))
    {
    TRACE1("CTabView:Failed to create view '%s'\n", pViewClass->m_lpszClassName);
    return -1;
    }

    CDocument* pDoc = GetDocument();
    if (pDoc != NULL)
    {
    ASSERT_VALID(pDoc);

    BOOL bFound = FALSE;
    for (POSITION pos = pDoc->GetFirstViewPosition(); !bFound && pos != NULL;)
    {
    if (pDoc->GetNextView(pos) == pView)
    {
    bFound = TRUE;
    }
    }

    if (!bFound)
    {
    pDoc->AddView(pView);
    }
    }

    m_wndTabs.InsertTab(pView, strViewLabel, iIndex);

    int nTabs = m_wndTabs.GetTabsNum();
    return nTabs - 1;
    }

    所以,只要AddView就可以,不再需要CCreateContext结构体作为参数,子tab就可以获得Doc指针了。



  • 相关阅读:
    linux sort根据日期时间排序方法记录
    gitlab数据迁移与升级记录
    ubuntu加压7z分卷
    docker环境运行elasticsearch以及汉化运行kibana
    nginx 403错误 检查nginx.conf user有没有问题,最好是当前用户
    系统属性file.encoding在JVM启动后,再次设置无法对系统的默认编码造成影响 & sun.jnu.encoding
    IDEA快速选择提示代码的设置
    log4j2可以打印到控制台,但无法打印到文件
    IDEA快捷键
    Eclipse自定义快捷键
  • 原文地址:https://www.cnblogs.com/zhangyonghugo/p/CTabView_MutiView.html
Copyright © 2020-2023  润新知