窗体创建完成,接下来我们讲讲控件的使用

首先在CFormView窗体下选项卡的成员变量,这里我选择MFC下的选项卡类库:CMFCTabCtrl

class CtabView : public CFormView
{
......
......
public:
CMFCTabCtrl m_tab; }

在初始化中动态创建选项卡并设置选项卡的样式,这里我们使用了Creat函数,他的原型是

BOOL Create(Style style, const RECT& rect, CWnd* pParentWnd, UINT nID, Location location = LOCATION_BOTTOM, BOOL bCloseBtn = FALSE);

第一个参数 Style style

是一个枚举类型,它定义了选项卡的样式,基本都为3D的样式

public:
    CMFCTabCtrl();

enum Style
    {
        STYLE_3D                      = 0,
        STYLE_FLAT                    = 1,
        STYLE_FLAT_SHARED_HORZ_SCROLL = 2,
        STYLE_3D_SCROLLED             = 3,
        STYLE_3D_ONENOTE              = 4,
        STYLE_3D_VS2005               = 5,
        STYLE_3D_ROUNDED              = 6,
        STYLE_3D_ROUNDED_SCROLL       = 7,
    };

第二个参数:设置选项卡大小

第三个参数:窗体句柄

第四个参数:设置选项卡控件ID

第五个参数:设置选项卡标签位置,MFC只给了上下两个方向来显示选项卡标签的位置

public:
 CMFCBaseTabCtrl();

enum Location
 {
  LOCATION_BOTTOM = 0,
  LOCATION_TOP = 1
 };

第六个参数:是否带关闭按钮,关闭按钮的逻辑是要自己实现的,这个我们下面再说。

添加选项卡代码:

void CtabView::OnInitialUpdate()
{
    CFormView::OnInitialUpdate();
    GetParentFrame()->RecalcLayout();
    ResizeParentToFit();     CRect rc;
    GetClientRect(rc);
    if (!m_tab.Create(CMFCTabCtrl::STYLE_3D_ONENOTE, rc, this, CMFCTabCtrl::RESIZE_HORIZ, CMFCTabCtrl::LOCATION_TOP))
    {
        return ;    
    }
}

创建好的选显卡是光秃秃的没有任何标签页的,这个需要我们手动添加

如何给选项卡创建标签页 ,这需要我们新建窗体在选项卡中显示

创建三个Dialog窗体备用

给窗体添加类,自动生成就可以

之后咱们给选项卡添加标签页,在选项卡中引用我们刚刚新建的Dialog类的指针

#include "Dialog1.h"
#include "Dialog2.h"
#include "Dialog3.h" class CtabView : public CFormView
{
public:
CMFCTabCtrl m_tab;
CDialog1 *m_dlg1;
CDialog2 *m_dlg2;
CDialog3 *m_dlg3; }

在程序初始化时动态创建的窗体

void CtabView::OnInitialUpdate()
{
 CFormView::OnInitialUpdate();
 GetParentFrame()->RecalcLayout();
 ResizeParentToFit();

CRect rc;
 GetClientRect(rc);
 if (!m_tab.Create(CMFCTabCtrl::STYLE_3D_ONENOTE, rc, this, 3000, CMFCTabCtrl::LOCATION_TOP,0))
 {
  return ;    
 }

m_dlg1=new CDialog1;
 m_dlg2=new CDialog2;
 m_dlg3=new CDialog3;

m_dlg1->Create(IDD_DIALOG1,&m_tab);
 m_dlg1->SetFont(&afxGlobalData.fontRegular);
 //m_dlg1->SetWindowTextW(L"Listctrl控件");

m_dlg2->Create(IDD_DIALOG2,&m_tab);
 m_dlg2->SetFont(&afxGlobalData.fontBold);
 //m_dlg2->SetWindowTextW(L"还没想好些什么控件");

m_dlg3->Create(IDD_DIALOG3,&m_tab);
 m_dlg3->SetFont(&afxGlobalData.fontDefaultGUIBold);
 /*m_dlg3->SetWindowTextW(L"还有什么控件比较难写吗");*/

m_tab.AddTab(m_dlg1,L"Listctrl控件");
 m_tab.AddTab(m_dlg2,L"还没想好些什么控件");
 m_tab.AddTab(m_dlg3,L"还有什么控件比较难写吗");

m_tab.EnableActiveTabCloseButton();//是否添加关闭选项卡按钮
 m_tab.SetActiveTab(0); //激活选项卡,以当前选项卡为第一页。

}

运行一下看一下效果

看起来还可以,但是当我们点击标签页的时候会发现,我们的窗体是弹出式的并不是嵌入式,这是因为我门少了少了一步,导致窗体在创建的时候是以弹出的方式打开。

打开窗体属性,将Style属性由弹出Popup改为Child子窗口,所有的标签用的窗体都要这么改

效果

里面的窗体太难看了了,把边框和标题去掉就好了

窗体属性,把Border改为None就好了

最终效果

代码 添加MFC选项卡.zip

 

如何优雅的写UI——(3)添加MFC选项卡的更多相关文章

  1. 如何优雅的写UI——(4)选项卡美化

    现在做出来的选项卡实在太丑的,咱们怎么把他弄得好看一点呢 tabctrl是可以添加icon图标的,那派生与tabctrl的mfctabctrl肯定也能添加图标,他们两个添加图标的原理一样,但是还是有点 ...

  2. 如何优雅的写UI——(5)选项卡功能实现

    先在我们的选项卡可以说能用了,每个标签页都能点进去,但是这还远远没到能用的地步,比如说你把窗口最大化后. 立马就露出马脚了,所以这篇我们要先讲讲tabctrl的最基本的功能实现 改变选项卡大小 上图的 ...

  3. 如何优雅的写UI——(1)MFC六大核心机制-程序初始化

    很多做软件开发的人都有一种对事情刨根问底的精神,例如我们一直在用的MFC,很方便,不用学太多原理性的知识就可以做出各种窗口程序,但喜欢钻研的朋友肯定想知道,到底微软帮我们做了些什么,让我们在它的框架下 ...

  4. 如何优雅的写UI——(2)MFC下基于CFormView的文档视图程序

    在MFC中可以创建多种类型的窗口程序,如对话框程序.单文档结构程序(非文档/视图结构).单文档(文档/视图结构)以及多文档视图结构程序等. 在编写一般的小工具时,我们的首选显然是对话框程序,不过基于对 ...

  5. 如何优雅的写UI——(6)内存泄漏

    控件讲了这么久,其实我的程序有两个Bug不知道大家有没有发现,这两个Bug都不会报错,对程序运行来说都没有阻碍,但是这种Bug对整个代码来说是一个很大的安全隐患. 什么是内存泄漏 内存泄漏(Memor ...

  6. 优雅的写好Vue项目代码 — 路由拆分、Vuex模块拆分、element按需加载

    目录 路由的拆分 VUEX模块拆分 Element UI库按需加载的优雅写法 路由的拆分 项目较大路由较多时,路由拆分是一个不错的代码优化方案,按不同业务分为多个模块,结构清晰便于统一管理. requ ...

  7. 关于如何写UI及屏幕适配的一些技巧

    因为公司开启了一个新的iOS项目, 所以近期比较忙, 没有更新博客,今天打算总结一下关于UI布局及屏幕适配的一些实战技巧,尤其使用纯代码,会对提升效率及代码易于维护等方面有明显帮助,这里提到的没有使用 ...

  8. 如何优雅的写一篇安利文-以Sugar ORM为例

    前言 我最近喜欢把写的十分优美的技术文章叫做安利文.首先,文章必须是原创而非软广:其次,阅读之后不仅能快速吸纳技术要点并入门开发,还能感同身受的体会作者热情洋溢的赞美和急于分享心得体验的心情,让人感觉 ...

  9. 【Xamarin挖墙脚系列:代码手写UI,xib和StoryBoard间的博弈,以及Interface Builder的一些小技巧(转)】

    正愁如何选择构建项目中的视图呢,现在官方推荐画板 Storybord...但是好像 xib貌似更胜一筹.以前的老棒子总喜欢装吊,用代码写....用代码堆一个HTML页面不知道你们尝试过没有.等页面做出 ...

随机推荐

  1. caffe for python (官方翻译)

    导言 本教程中,我们将会利用Caffe官方提供的深度模型——CaffeNet(该模型是基于Krizhevsky等人的模型的)来演示图像识别与分类.我们将分别用CPU和GPU来进行演示,并对比其性能.然 ...

  2. ajax发送请求的数据类型

    1.如果要传给后台的是json形式的数据 2.如果要传给后台的是formdata形式的数据

  3. NodeJS学习笔记 (21)事件机制-events(ok)

    模块概览 events模块是node的核心模块之一,几乎所有常用的node模块都继承了events模块,比如http.fs等. 模块本身非常简单,API虽然也不少,但常用的就那么几个,这里举几个简单例 ...

  4. luogu P3157 [CQOI2011]动态逆序对(CDQ分治)

    题目描述 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的任务是在每次删除一个元素之前统计整个序列的逆序 ...

  5. python 基础使用list、dict、set、可变与不可变对象

    参考链接:https://www.liaoxuefeng.com/wiki/1016959663602400/1017104324028448 dict是字典,可以储存键值对类型的值,set与dict ...

  6. Unity 获得Android Context上下文

    1.获取Context AndroidJavaObject context = new AndroidJavaClass ("com.unity3d.player.UnityPlayer&q ...

  7. POJ——T 1988 Cube Stacking

    http://poj.org/problem?id=1988 Time Limit: 2000MS   Memory Limit: 30000K Total Submissions: 25865   ...

  8. vdceye 最新中文界面

    最新的vdceye 的界面.左边菜单增加了问题.并增加了虚拟摄像机部分 watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvdmlkZW9fZGM=/font/5 ...

  9. android 在短信发送界面, 短信发送失败时,提示音不完整,会被中断

    1. 当一条SMS到来, 此时SMS是unseen状态, 就会弹出Notification提示用户 2. 但假设处于同一个联系人的界面下, 用户会立马看到这条SMS, 此时这条SMS会被高速的标记为s ...

  10. Codeforces Round #313 C. Gerald&#39;s Hexagon(放三角形)

    C. Gerald's Hexagon time limit per test 2 seconds memory limit per test 256 megabytes input standard ...