如果你用 MFC 编写过多文档界面(MDI)Windows 程序,那么肯定知道:如果父窗口标题为“PCaption”,子窗口标题为“CCaption”,那么每当子窗口最大化并处于激活状态时,子窗口标题一般都会与父窗口标题合二为一,变成“PCaption-[CCaption]”。 这是一种 MDI 的默认行为。用 C# 编写多文档界面程序也不例外。很多用户都不喜欢这种缺省特性,往往想用定制的窗口标题取而代之。本文将示范如何在C#程序中定制和修改MDI应用的窗口标题。
如果用MFC来编程,只要改写框架窗口类的虚函数 CFrameWnd::OnUpdateFrameTitle 即可。那么在微软的 .NET 框架中如何用C#实现相同的功能呢?首先,我们必须理解 MDI 本身是如何通过 Windows 核心 API 来实现自己的行为特性的,其实这与MFC或者.NET的公共语言运行时(CLR)没有什么关系。在创建MDI应用时,框架及其子窗口有各自专门的窗口过程,DefFrameProc 和 DefMDIChildProc,一个处理各种 WM_MDIXXX 消息以及其它类似 WM_SIZE, WM_SYSCOMMAND 的消息,另一个实现 MDI 行为。
如果用纯 C 代码编写,那么必须自己负责用 DefFrameProc 和 DefMDIChildProc 创建窗口;在 MFC 中则使用 CMDIFrameWnd/CMDIChildWnd;.NET 框架平台里则设置 Form.IsMdiContainer 和 Form.MdiParent,不管用哪种方式,其核心都是 user kernel,尤其是 DefFrameProc,当 MDI 子窗口最大化时,它会联接父子窗口的标题文本来产生主窗口标题串。理解了这一点,下面我来示范如何改写MDI。这个例子的原始版本来自 MSDN 库中用C#写的 Scribble MDI(用 “scribble sample”搜索一下即可找到)。基本思路是首先在 Scribble 例子的 MainWindow 中改写 WM_GETTEXT 消息处理例程,必须添加两个数据成员:NormalText 和 MaximizedText,用它们来保存常态和最大化状态的标题 :
// in Scribble.cs, MainWindow class private String NormalText = "Scribble2"; private String MaximizedText = "Window is now maximized"; |
如果想让其它类存取这两个成员,那么可以通过属性机制代替数据成员,如:
private String normaltext; public String NormalText { get { return normaltext; } set { normaltext = value; } } |
因为在例子程序中 MainWindow 是唯一一个存取该字符串的类,所以没有必要使用属性机制。有了这两个新的数据成员,你要做的只是 改写 WM_GETTEXT 处理例程,返回子窗口最大化状态以及常态时的标题文本。那么如何改写 WM_GETTEXT 处理例程呢?
精品教程尽在www.xvna.com
Windows.Forms 提供了一些 处理 WM_XXX 消息的虚拟函数,如 OnResize/WM_SIZE等,但是恰恰缺少与 WM_GETTEXT 相关东东(OnGetText/WM_GETTEXT)。不要担心,没有虚函数,我们总是可以改写包罗万象的 WndProc 处理例程。为此必须知道所处理的消息ID,也就是 WM_GETTEXT 的消息 ID = 0x000D,有人会问,你是怎么知道这个消息的 ID 是 0x000D 啊,很简单,一种方法是运行 SPY 获取,另一种方法是直接查找Windows SDK 中的 winuser.h 头文件。一旦你能深入到 WndProc 这一层次编写代码,那么你基本上能用 C 语言写程序了,因为 Win32 API 和其它语言之间所有东东通过 WPARAMs 和 LPARAMs 参数传递的,包括字符串在内。对于 WM_GETTEXT 来说,Message.LParam 是指向 char* 的指针,Message.WParam 是该指针长度。也就是说你必须完成将文本串拷贝到调用者的缓冲里。好在这并不是太难,下面是程序代码:
public class MainWindow : System.Windows.Forms.Form { private String NormalText = "Scribble2"; private String MaximizedText = "Window is now maximized"; // Handle WM_GETTEXT: Return maximized or // normal text, depending on // state of active MDI child window. protected override void WndProc(ref Message m) { const int WM_GETTEXT = 0x000D; if (m.Msg==WM_GETTEXT) { Form active = this.ActiveMdiChild; String s = active!=null && active.WindowState==FormWindowState.Maximized ? MaximizedText : NormalText; char[] c = s.ToCharArray(); IntPtr buf = m.LParam; int len = c.Length; Marshal.Copy(c, 0, buf, Math.Min((int)m.WParam, len)); m.Result = (IntPtr)len; return; } base.WndProc(ref m); } ...... // rest of MainWindow unchanged from Scribble sample } |
经过上述的改动,现在运行程序,当MDI子窗口最大化时,主窗口标题显示的文本是“Window is now maximized”,如图一所示,
图一 子窗口最大化时的主窗口标题 |
当两个窗口处于常态时,其画面如图二所示:
图二 子窗口在常态时两个窗口的标题 |