MFC Frame-Splitter模型实例原理解析

1. 新建一个SDI工程,去掉“Document/View architecture support”

2. 添加 CTreeView 类的派生类 CLeftView 作为分隔器的左侧视图,添加 CListView 的派生类 CRightView 作为右边视图。可能需要 #include <afxcview.h>

3. 在CLeftView类中添加成员变量 CTreeCtrl& m_tree; 在CRightView类中添加成员变量 CListCtrl& m_list; 并在构造函数中初始化

CLeftView::CLeftView() :m_tree(GetTreeCtrl())
{
}
CRightView::CRightView() :m_list(GetListCtrl())
{
}

4. 在主框架类 CMainFrame 类的头文件中,去掉 CChildView 类变量,添加 CSplitterWnd m_split;CLeftView* m_pLeftView;CRightView* m_pRightView;在CMainFrame的cpp文件中去掉CChildView的相关代码。

在CMainFrame类中添加虚函数OnCreateClient,该函数是在基类函数 CFrameWnd::OnCreate 执行的过程中回调OnCreateClient函数,主要用于创建视图类等客户区子窗口。

在CLeftView类中添加虚函数OnInitUpdate,该函数创建类之后第一个进入的函数,类似对话框的OnInitDialog函数。再添加TVN_SELCHANGEED消息映射函数,并修改代码调用CRightView的Refresh函数。

在CRightView类中添加虚函数OnInitUpdate,添加普通成员函数 Refresh。

代码如下:

MainFrm.cpp

// MainFrm.cpp : CMainFrame 类的实现
//

#include "stdafx.h"
#include "sp.h"

#include "MainFrm.h"

#ifdef _DEBUG
#define new DEBUG_NEW
#endif

// CMainFrame

IMPLEMENT_DYNAMIC(CMainFrame, CFrameWnd)

BEGIN_MESSAGE_MAP(CMainFrame, CFrameWnd)
  ON_WM_CREATE()
  ON_WM_SETFOCUS()
END_MESSAGE_MAP()

static UINT indicators[] =
{
  ID_SEPARATOR,      // 状态行指示器
  ID_INDICATOR_CAPS,
  ID_INDICATOR_NUM,
  ID_INDICATOR_SCRL,
};

// CMainFrame 构造/析构

CMainFrame::CMainFrame()
{
  // TODO: 在此添加成员初始化代码
}

CMainFrame::~CMainFrame()
{
}

BOOL CMainFrame::OnCreateClient(LPCREATESTRUCT lpcs, CCreateContext* pContext)
{
  // TODO: 在此添加专用代码和/或调用基类
  if (!m_split.CreateStatic(this, 1, 2))
    return FALSE;
  m_split.CreateView(0, 0, RUNTIME_CLASS(CLeftView), CSize(150, 0), NULL);
  m_split.CreateView(0, 1, RUNTIME_CLASS(CRightView), CSize(0, 0), NULL);
  m_pLeftView = (CLeftView*)m_split.GetPane(0, 0);
  m_pRightView = (CRightView*)m_split.GetPane(0, 1);
  return TRUE;

  //  return CFrameWnd::OnCreateClient(lpcs, pContext);
}

int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
  if (CFrameWnd::OnCreate(lpCreateStruct) == -1)
    return -1;

  // 创建一个视图以占用框架的工作区
  //if (!m_wndView.Create(NULL, NULL, AFX_WS_DEFAULT_VIEW, CRect(0, 0, 0, 0), this, AFX_IDW_PANE_FIRST, NULL))
  //{
  //  TRACE0("未能创建视图窗口\n");
  //  return -1;
  //}

  if (!m_wndToolBar.CreateEx(this, TBSTYLE_FLAT, WS_CHILD | WS_VISIBLE | CBRS_TOP | CBRS_GRIPPER | CBRS_TOOLTIPS | CBRS_FLYBY | CBRS_SIZE_DYNAMIC) ||
    !m_wndToolBar.LoadToolBar(IDR_MAINFRAME))
  {
    TRACE0("未能创建工具栏\n");
    return -1;   // 未能创建
  }

  if (!m_wndStatusBar.Create(this))
  {
    TRACE0("未能创建状态栏\n");
    return -1;   // 未能创建
  }
  m_wndStatusBar.SetIndicators(indicators, sizeof(indicators)/sizeof(UINT));

  // TODO: 如果不需要可停靠工具栏,则删除这三行
  m_wndToolBar.EnableDocking(CBRS_ALIGN_ANY);
  EnableDocking(CBRS_ALIGN_ANY);
  DockControlBar(&m_wndToolBar);

  return 0;
}

BOOL CMainFrame::PreCreateWindow(CREATESTRUCT& cs)
{
  if( !CFrameWnd::PreCreateWindow(cs) )
    return FALSE;
  // TODO: 在此处通过修改
  // CREATESTRUCT cs 来修改窗口类或样式

  cs.dwExStyle &= ~WS_EX_CLIENTEDGE;
  cs.lpszClass = AfxRegisterWndClass(0);

  return TRUE;
}

// CMainFrame 诊断

#ifdef _DEBUG
void CMainFrame::AssertValid() const
{
  CFrameWnd::AssertValid();
}

void CMainFrame::Dump(CDumpContext& dc) const
{
  CFrameWnd::Dump(dc);
}
#endif //_DEBUG

// CMainFrame 消息处理程序

void CMainFrame::OnSetFocus(CWnd* /*pOldWnd*/)
{
  // 将焦点前移到视图窗口
//  m_wndView.SetFocus();
  m_pLeftView->SetFocus();
}

BOOL CMainFrame::OnCmdMsg(UINT nID, int nCode, void* pExtra, AFX_CMDHANDLERINFO* pHandlerInfo)
{
  // 让视图第一次尝试该命令
  //if (m_wndView.OnCmdMsg(nID, nCode, pExtra, pHandlerInfo))
  //  return TRUE;
  CWnd* pWnd = GetFocus();
  if (pWnd && pWnd != this && pWnd->OnCmdMsg(nID, nCode, pExtra, pHandlerInfo))
  {
    return TRUE;
  }

  // 否则,执行默认处理
  return CFrameWnd::OnCmdMsg(nID, nCode, pExtra, pHandlerInfo);
}

LeftView.cpp

// LeftView.cpp : 实现文件
//

#include "stdafx.h"
#include "sp.h"
#include "LeftView.h"

// CLeftView

IMPLEMENT_DYNCREATE(CLeftView, CTreeView)

CLeftView::CLeftView() :m_tree(GetTreeCtrl())
{

}

CLeftView::~CLeftView()
{
}

BEGIN_MESSAGE_MAP(CLeftView, CTreeView)
  ON_NOTIFY_REFLECT(TVN_SELCHANGED, &CLeftView::OnTvnSelchanged)
END_MESSAGE_MAP()

// CLeftView 诊断

#ifdef _DEBUG
void CLeftView::AssertValid() const
{
  CTreeView::AssertValid();
}

#ifndef _WIN32_WCE
void CLeftView::Dump(CDumpContext& dc) const
{
  CTreeView::Dump(dc);
}
#endif
#endif //_DEBUG

// CLeftView 消息处理程序

void CLeftView::OnInitialUpdate()
{
  CTreeView::OnInitialUpdate();

  // TODO: 在此添加专用代码和/或调用基类
  if (m_tree.GetChildItem(NULL))
    return;
  m_tree.ModifyStyle(0, TVS_HASLINES | TVS_LINESATROOT | TVS_HASBUTTONS | TVS_SHOWSELALWAYS);
  HTREEITEM hItem = m_tree.InsertItem(_T("我的电脑"));
  char buf[100];
  DWORD len = GetLogicalDriveStringsA(sizeof(buf) / sizeof(buf[0]), buf);//获取的buf,例如c:\<null>d:\<null><null>
  char *p = buf;
  CString str;
  while (*p)
  {
    str = p;
    int nLen = str.GetLength();
    if (nLen < 1)
      break;
    if ('\\' == str[nLen - 1])
      str = str.Left(nLen - 1);
    m_tree.InsertItem(str, hItem);
    p += nLen + 1;
  }
  m_tree.Expand(hItem, TVE_EXPAND);
}

#include "MainFrm.h"
void CLeftView::OnTvnSelchanged(NMHDR *pNMHDR, LRESULT *pResult)
{
  LPNMTREEVIEW p = reinterpret_cast<LPNMTREEVIEW>(pNMHDR);
  // TODO: 在此添加控件通知处理程序代码
  HTREEITEM hItem = p->itemNew.hItem;
  CString szPath;
  if (hItem != m_tree.GetRootItem())
    szPath = m_tree.GetItemText(hItem);
  CMainFrame* pFrame = (CMainFrame*)AfxGetMainWnd();
  pFrame->m_pRightView->Refresh(szPath);

  *pResult = 0;
}

RightView.cpp

// RightView.cpp : 实现文件
//

#include "stdafx.h"
#include "sp.h"
#include "RightView.h"

// CRightView

IMPLEMENT_DYNCREATE(CRightView, CListView)

CRightView::CRightView() :m_list(GetListCtrl())
{

}

CRightView::~CRightView()
{
}

BEGIN_MESSAGE_MAP(CRightView, CListView)
END_MESSAGE_MAP()

// CRightView 诊断

#ifdef _DEBUG
void CRightView::AssertValid() const
{
  CListView::AssertValid();
}

#ifndef _WIN32_WCE
void CRightView::Dump(CDumpContext& dc) const
{
  CListView::Dump(dc);
}
#endif
#endif //_DEBUG

// CRightView 消息处理程序

void CRightView::OnInitialUpdate()
{
  CListView::OnInitialUpdate();

  // TODO: 在此添加专用代码和/或调用基类
  if (m_list.GetStyle() & LVS_REPORT)
    return;
  m_list.ModifyStyle(0, LVS_REPORT);
  m_list.InsertColumn(0, _T("名称"), 0, 120);
  m_list.InsertColumn(1, _T("大小"), 0, 120);
  m_list.InsertColumn(2, _T("类型"), 0, 120);
  m_list.InsertColumn(3, _T("修改日期"), 0, 120);
}

void CRightView::Refresh(CString szPath)
{
  m_list.DeleteAllItems();
  if (szPath.IsEmpty())
    return;
  CFileFind ff;
  BOOL b = ff.FindFile(szPath + _T("/*.*"));
  int i = 0;
  while (b)
  {
    b = ff.FindNextFile();
    if (ff.IsDots())
      continue;
    m_list.InsertItem(i, ff.GetFileName());
    if (ff.IsDirectory())
      m_list.SetItemText(i, 2, _T("文件夹"));
    else
    {
      int nLen = ff.GetLength() / 1024;
      if (!nLen)
        nLen = 1;
      szPath.Format(_T("%d kb"), nLen);
      m_list.SetItemText(i, 1, szPath);
      m_list.SetItemText(i, 2, _T("文件"));
    }
    FILETIME ft;
    ff.GetLastWriteTime(&ft);
    COleDateTime time(ft);
    szPath.Format(_T("%d-%02d-%02d %02d:%02d:%02d"),
      time.GetYear(), time.GetMonth(), time.GetDay(),
      time.GetHour(), time.GetMinute(), time.GetSecond());
    m_list.SetItemText(i, 3, szPath);
    ++i;
  }
}

运行结果:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • mfc文件操作CFile类之创建文件的方法

    mfc文件操作类CFile CFile类的几种构造函数 CFile();//不带参数的构造函数. CFile(int hFile);//带文件句柄参数的构造函数. CFile( LPCTSTR lpszFileName,//要打开的文件路径 UINT nOpenFlage //打开文件时的共享和访问模式 );//带文件路径和打开方式的构造函数 除了用构造函数打开文件外还可以用CFile类的成员函数Open来打开 OPen函数原型 virtual BOOL Open( LPCTSTR lpszFi

  • VS2019安装配置MFC(安装vs2019时没有安装mfc)

    1.查看vs2019里面有没有配置MFC 新建项目,如果能在右边找到MFC应用则配置了,否则没有 2.*如果没有安装,将右边的的竖直移动条拉到底部,点击安装多个工具和供能* 3.选中在C++桌面开发==>在右侧选中相关功能==>点击修改 到此这篇关于VS2019安装配置MFC(安装vs2019时没有安装mfc)的文章就介绍到这了,更多相关VS2019安装配置MFC内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

  • 如何使用visual studio2019创建简单的MFC窗口(使用C++)

    本文介绍了如何使用visual studio2019创建简单的MFC窗口(使用C++) ```cpp 使用visual studio 2019 创建过程请参考Bili的上一篇文章⬇⬇ →!使用visual studio 2019 创建简单的MFC窗口「使用底层的C语言」 #include<windows.h> //底层实现窗口的头文件 //6.处理窗口过程 //CALLBACK 代表_stdcall 参数的传递顺序:从右到左依次入栈,并且函数返回前清空堆栈 LRESULT CALLBACK W

  • 梅尔频率倒谱系数(mfcc)及Python实现

    语音识别系统的第一步是进行特征提取,mfcc是描述短时功率谱包络的一种特征,在语音识别系统中被广泛应用. 一.mel滤波器 每一段语音信号被分为多帧,每帧信号都对应一个频谱(通过FFT变换实现),频谱表示频率与信号能量之间的关系.mel滤波器是指多个带通滤波器,在mel频率中带通滤波器的通带是等宽的,但在赫兹(Hertz)频谱内mel滤波器在低频处较密集切通带较窄,高频处较稀疏且通带较宽,旨在通过在较低频率处更具辨别性并且在较高频率处较少辨别性来模拟非线性人类耳朵对声音的感知. 赫兹频率和梅尔频

  • MFC实现字幕滚动效果

    本文实例为大家分享了MFC实现字幕滚动效果的具体代码,供大家参考,具体内容如下 1.创建对话框工程 这一步很简单,可以参考我之前的博客.这里可以先放出我程序运行的效果图.如果不是你所需要的可以直接关闭博客了. 2.添加控件 在界面上面主要用到的就是CStatic, Cedit,Cbutton这里给我我对话框的资源图: 3.添加消息 需要添加定时消息,背景颜色消息,添加消息: ON_WM_CTLCOLOR() ON_WM_TIMER() HBRUSH CDemoLEDDlg::OnCtlColor

  • VS2010+Opencv+MFC读取图像和视频显示在Picture控件

    VS2010+Opencv+MFC读取图像和视频显示在Picture控件,供大家参考,具体内容如下 1.新建MFC对话框应用程序. 其余选项默认,单击完成,创建出对话框应用程序.删掉原来自带的一些控件,添加picture控件和两个按钮. 2.由于以后的代码会用到CvvImage类,而opencv2.3以后就去掉了对它的支持,这里先介绍添加CvvImage支持的方法,直接能用的可以略过这一步. 如下图添加相应的文件: 这里附上两个文件的源码方便使用. #pragma once #ifndef CV

  • VS2019以及MFC的安装详细教程

    本文为大家分享了VS2019以及MFC的安装详细教程,供大家参考,具体内容如下 一.安装过程: 1.搜索visual studio 进入官网. 2.选择community 2019下载. 3.现在完成后点击安装,等待安装完成 4.自动弹出一下界面,工作负载这里勾选使用C++的桌面开发和Visual Studio 扩展开发.另外,在使用C++的桌面开发选中后的右侧安装详细信息栏中勾选适用于最新的v142生成工具的C++MFC(x86和x64)这一项.选中后点击安装.(一共勾选3个地方,如下图) 5

  • MFC Frame-Splitter模型实例原理解析

    1. 新建一个SDI工程,去掉"Document/View architecture support" 2. 添加 CTreeView 类的派生类 CLeftView 作为分隔器的左侧视图,添加 CListView 的派生类 CRightView 作为右边视图.可能需要 #include <afxcview.h> 3. 在CLeftView类中添加成员变量 CTreeCtrl& m_tree; 在CRightView类中添加成员变量 CListCtrl& m

  • Java枚举类接口实例原理解析

    这篇文章主要介绍了Java枚举类接口实例原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 枚举类可以实现一个或多个接口.与普通类实现接口完全一样,枚举类实现接口时,需要实现该接口所包含的方法. 如果需要每个枚举值在调用同一个方法时呈现不同的行为,则可以让每个枚举值在{...}匿名块中实现自己的业务逻辑. public interface IGradeHandler { String getGrade(String studentName)

  • Java内存模型原子性原理及实例解析

    这篇文章主要介绍了Java内存模型原子性原理及实例解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 本文就具体来讲讲JMM是如何保证共享变量访问的原子性的. 原子性问题 原子性是指:一个或多个操作,要么全部执行且在执行过程中不被任何因素打断,要么全部不执行. 下面就是一段会出现原子性问题的代码: public class AtomicProblem { private static Logger logger = LoggerFactory.

  • Laravel框架源码解析之模型Model原理与用法解析

    本文实例讲述了Laravel框架源码解析之模型Model原理与用法.分享给大家供大家参考,具体如下: 前言 提前预祝猿人们国庆快乐,吃好.喝好.玩好,我会在电视上看着你们. 根据单一责任开发原则来讲,在laravel的开发过程中每个表都应建立一个model对外服务和调用.类似于这样 namespace App\Models; use Illuminate\Database\Eloquent\Model; class User extends Model { protected $table =

  • RabbitMQ简单队列实例及原理解析

    这篇文章主要介绍了RabbitMQ简单队列实例及原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 RabbitMQ 简述 RabbitMQ是一个消息代理:它接受并转发消息. 您可以将其视为邮局:当您将要把寄发的邮件投递到邮箱中时,您可以确信Postman 先生最终会将邮件发送给收件人. 在这个比喻中,RabbitMQ是一个邮箱,邮局和邮递员,用来接受,存储和转发二进制数据块的消息. 队列就像是在RabbitMQ中扮演邮箱的角色. 虽然消息

  • Python中使用gflags实例及原理解析

    这篇文章主要介绍了Python中使用gflags实例及原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 安装命令pip install python-gflags 使用示例: import gflags FLAGS = gflags.FLAGS gflags.DEFINE_string('name', 'ming', 'this is a value') gflags.DEFINE_integer('qps', 0, 'test qps'

  • Python爬虫解析网页的4种方式实例及原理解析

    这篇文章主要介绍了Python爬虫解析网页的4种方式实例及原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 用Python写爬虫工具在现在是一种司空见惯的事情,每个人都希望能够写一段程序去互联网上扒一点资料下来,用于数据分析或者干点别的事情.​ 我们知道,爬虫的原理无非是把目标网址的内容下载下来存储到内存中,这个时候它的内容其实是一堆HTML,然后再对这些HTML内容进行解析,按照自己的想法提取出想要的数据,所以今天我们主要来讲四种在Py

  • Python While循环语句实例演示及原理解析

    这篇文章主要介绍了Python While循环语句实例演示及原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 Python 编程中 while 语句用于循环执行程序,即在某条件下,循环执行某段程序,以处理需要重复处理的相同任务.其基本形式为: while 判断条件: 执行语句-- 执行语句可以是单个语句或语句块.判断条件可以是任何表达式,任何非零.或非空(null)的值均为true. 当判断条件假false时,循环结束. 执行流程图如下:

  • Python @property原理解析和用法实例

    这篇文章主要介绍了Python @property原理解析和用法实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 在我们定义数据库字段类的时候,往往需要对其中的类属性做一些限制,一般用get和set方法来写,那在python中,我们该怎么做能够少写代码,又能优雅的实现想要的限制,减少错误的发生呢,这时候就需要我们的@property闪亮登场啦,巴拉巴拉能量--.. 用代码来举例子更容易理解,比如一个学生成绩表定义成这样 class Stude

  • Pytest mark使用实例及原理解析

    这篇文章主要介绍了Pytest mark使用实例及原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 使用方法: 1.注册标签名 2.在测试用例/测试类前面加上:@pytest.mark.标签名 打标记范围:测试用例.测试类.模块文件 注册方式: 1.单个标签: 在conftest.py添加如下代码: def pytest_configure(config): # demo是标签名 config.addinivalue_line("mark

随机推荐