【Windows编程】创建多文档界面
前面我们所举的例子中都是单文档界面框架,也就是说这个窗口里面的客户区就是一个文档界面,可以编写程序在里面输入或者绘制文本和图形输出,但是不能有出现多个文档的情况。比如下面的UltraEdit就是一个典型的多文档界面,他可以同时编辑多个文档,每个文档还可以最大化,最小化等等,我们今天就来看看多文档的基本框架是怎么实现的。
多文档界面框架创建过程需要以下几步:
- 主框架窗口创建
主框架窗的创建跟普通的窗口没有什么区别,就是自己注册一个类并用该类创建一个重叠窗口,这个可以用CreateWindow/CreateWindowEx函数完成,主框架窗口可以用自己的菜单和状态栏。
- 客户区窗口创建
客户区创建的创建同样用你CreateWindow,但需要指定类为“MDICLIENT”,用这个类会创建多文档的客户区窗口;或者采用CreateWindowEx函数,指定扩展风格为WS_EX_MDICHILD,客户区不需要菜单和状态栏。
- 视图窗口创建
创建工作或者视图窗口作为实际文档窗口,这个也是需要自己注册类并创建自己需要的视图窗口。视图窗口可以有自己的菜单,一般不需要状态栏。当视图窗口激活时,主窗口的菜单需要替换成视图窗口的菜单,因为这个时候往往是对视图窗口进行操作,菜单替换也是理所当然的。菜单设置通过WM_MDISETMENU消息完成。该消息定义如下:
SendMessage(hWndControl, WM_MDISETMENU, wParam, lParam)
wParam表示新框架窗口菜单,
lParam表示要设置的菜单句柄
OK,基本的API就这些,下面我们直接上demo程序来演示,由于视图一般是一个多文档的实例,所以demo代码中用类来存储视图,每一个视图就是一个类的实例:
限于内容浏览限制,这里只贴出关键部分代码,查看完整的代码请猛戳左下角的“阅读原文”。
主文件,该文件包括主框架窗口、客户窗口以及视图窗口调用:
ghWndMainFrame = CreateWindow(szMDIMainFrame, gszAppName, WS_OVERLAPPEDWINDOW | WS_CLIPCHILDREN, CW_USEDEFAULT, // allows system choose an x position CW_USEDEFAULT, // allows system choose a y position 400, 300, NULL, ghMainFrameMenu, hInstance, NULL);
LRESULT CALLBACK MainFrameProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { switch(message) { case WM_CREATE: { CLIENTCREATESTRUCT ccs; ccs.hWindowMenu = NULL; ccs.idFirstChild = IDM_FIRSTCHILD; ghWndMDIClient = CreateWindow(TEXT("MDICLIENT"), NULL, WS_CHILD | WS_CLIPCHILDREN | WS_VISIBLE, 0, 0, 0, 0, hWnd, NULL, g_hInst, (void *)&ccs); #ifdef CLIENT_WND_BGROUND pfOrigClientWndProc = (WNDPROC)SetWindowLong(ghWndMDIClient, GWL_WNDPROC, (LONG)ClientWndSubClassProc); #endif } return 0; case WM_COMMAND: switch(LOWORD(wParam)) { case IDM_FILE_NEW: CreateDocView(ghWndMDIClient, g_hInst); return 0; } break; case WM_DESTROY: PostQuitMessage(0); return 0; } return DefFrameProc(hWnd, ghWndMDIClient, message, wParam, lParam); }
LRESULT CALLBACK ViewWindowProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { switch(message) { case WM_MDIACTIVATE: { HWND hWndClient = GetParent(hWnd); HWND hWndFrame = GetParent(hWndClient); HMENU hMenu = GetSubMenu(ghChildWndMenu, 0); if (lParam == (LPARAM)hWnd) { SendMessage(hWndClient, WM_MDISETMENU, (WPARAM)ghChildWndMenu, (LPARAM)hMenu); EnableMenuItem(ghChildWndMenu, IDM_EDIT_COPY, MF_GRAYED); } // Set the framewindow menu if losing focus if (lParam != (LPARAM)hWnd) { SendMessage(hWndClient, WM_MDISETMENU, (WPARAM)ghMainFrameMenu, (LPARAM)NULL); } // call DrawMenuBar after the menu items are set DrawMenuBar(hWndFrame); return 0 ; } case WM_CLOSE: delete g_pSystemInfo; break; case WM_DESTROY: return 0; } return DefMDIChildProc(hWnd, message, wParam, lParam); //Frame window calls DefFrameProc rather than DefWindowProc } HWND CViewInfo::CreateViewWindow(HWND hParentWnd) { WNDCLASSEX wcDoc; wcDoc.cbSize = sizeof(WNDCLASSEX); wcDoc.style = CS_HREDRAW | CS_VREDRAW; wcDoc.lpfnWndProc = ViewWindowProc; wcDoc.cbClsExtra = 0; wcDoc.cbWndExtra = 0; wcDoc.hInstance = m_hInst; wcDoc.hIcon = NULL; wcDoc.hCursor = LoadCursor(NULL, IDC_ARROW); wcDoc.hbrBackground = (HBRUSH) GetStockObject(WHITE_BRUSH); wcDoc.lpszMenuName = NULL; wcDoc.lpszClassName = szViewWndName; wcDoc.hIconSm = NULL; if(!RegisterClassEx(&wcDoc)) { DWORD dw_LastError = GetLastError(); if(ERROR_CLASS_ALREADY_EXISTS != dw_LastError) { return 0; } } m_hWndView = CreateWindowEx(WS_EX_MDICHILD, szViewWndName, TEXT("New view"), 0, CW_USEDEFAULT, CW_USEDEFAULT, 300, 200, hParentWnd, NULL, m_hInst, NULL); return m_hWndView; } HWND CreateDocView(HWND hParentWnd, HINSTANCE hInstance) { HWND hView; g_pSystemInfo = new CViewInfo(hInstance); hView = g_pSystemInfo->CreateViewWindow(hParentWnd); if (hView == NULL) { delete g_pSystemInfo; g_pSystemInfo = NULL; } return hView; } void CreateChildWndMenu(void) { //View菜单 ghChildWndMenu = CreateMenu(); //编辑菜单 HMENU hEditMenu = CreateMenu(); AppendMenu(hEditMenu, MF_STRING, IDM_EDIT_COPY, TEXT("&Copy")); AppendMenu(hEditMenu, MF_STRING, IDM_EDIT_PASTE, TEXT("&Paste")); AppendMenu(ghChildWndMenu, MF_POPUP, (UINT_PTR)hEditMenu, TEXT("&Edit")); }
本实例运行后,界面如下:
选择File->New新建一个视图后demo程序如下,可以看到菜单编程视图的菜单:
最大化后可以看到视图窗口和填满客户窗口:
虽然本程序只实现了一个视图的实例,但是再增加一个是很容易的,只要想办法在菜单中调用CreateDocView函数,并且正确处理对象指针即可。实例并没有增加状态栏,因为这个对多文档并不是必须的,要增加的读者可以参考前面的创建Toolbar和Statusbar一文。
本实例实现了一个基本的多文档窗口框架,读者朋友可以在此基础上加上工具栏、状态栏、视图窗口创建对类的处理,多实例以及具体的需求,完成实用化的多文档界面。
- (33) Joda-Time / 计算机程序的思维逻辑
- Python实现守护进程
- 初探Anaconda——最省心的Python版本和第三方库管理
- Linux环境下JDK/Eclipse一键安装脚本
- (31) 剖析Arrays / 计算机程序的思维逻辑
- 应用自然语言处理(NLP)解码电影
- 不引入新的数组,实现数组元素交换位置函数
- (30) 剖析StringBuilder / 计算机程序的思维逻辑
- Java初始化顺序
- ConcurrentHashMap使用示例
- (40) 剖析HashMap / 计算机程序的思维逻辑
- nginx配置https(亲测可用)
- linux中无 conio.h的解决办法
- 运用适配器模式应对项目中的变化
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- fpga实现YCbCr444转RGB
- UML类图自动生成,太爽了
- Python爬虫之mongodb介绍和安装
- 一句话说清楚 CountDownLatch 和 CyclicBarrier 的区别
- Android 功耗(8)---如何找到阻止进入deep idle SODI的元凶
- 【技术创作101训练营】三种不同场景下 vue 组件动态加载的方法及实现
- leetcode链表之回文链表
- 如何理解 Go 中的反射
- Synchronized深入分析
- Spring的一些零碎知识点整理
- CentOS7上安装并配置KVM,以及通过KVM安装CentOS系统
- 建议收藏 哭着喊着 从C语言转向C++刷算法
- Spring的事务管理
- 面向切面的Spring
- 搭建ELK日志分析平台(下)—— 搭建kibana和logstash服务器