wpf usercontrol窗口怎么使用windowproc

wpf usercontrol窗口怎么使用windowproc,第1张

在学MFC 的第一课就是讲windows的消息机制,其中这一部分是通过WndProc这个函数来完成的,用一个消息循环来捕捉消息。虽然在WPF里 没有 WndProc 这个函数,但是提供了另一种方法来完成挂钩的功能。在WinForm 里 只需重载 WndProc 函数便可完成消息响应。

WinForm:

private const int WM_SIZEING = 0x0214 //窗口改变大小消息

protected override void WndProc(ref System.Windows.Forms.Message m )

{

switch(m.Msg)

{

case WM_SIZEING:

//...........你的代码

break

default:

base.WndProc(ref m) // 调用基类函数处理其他消息。

break

}

}

WPF:

在你运行WPF程序的时候——new App().Run(new MyWindow()),有时候要使用Win32API处理消息的时候该怎么办?其实很简单,一切的需要都在System.Windows.Interop这个namespace中,这个namespace中所包含的东西主要是为 Windows Presentation Foundation (WPF) 和其他技术(如 Win32API)之间的互 *** 作提供支持类型,并为涉及 WPF 的其他特定互 *** 作方案提供基类。其他的我就不加以说明了,这里的主题是如何在WPF中使用WindowProc处理消息。

在WPF中使用WindowProc,首先要获得一个HWND句柄(一个IntPtr类型),然后通过创建一个HwndSource对象,再通过调用这个对象中的AddHook方法来添加WindowProc。

首先看看HwndSource ,msdn是这么写的:

HwndSource 实现包含 WPF 内容的 Win32 窗口。 WPF 内容在此窗口中排列、度量、呈现,并且可交互式输入。

HwndSource 类设计用于一般的交互 *** 作,而不是设计用作托管 HWND 包装。 通常,它不会提供 *** 作窗口的托管方法或检测其状态的属性。 相反,HwndSource 类提供通过 Handle 属性对 Win32 窗口句柄 (HWND) 的访问,可通过 PInvoke 技术将其传递到 Win32 APIs 以 *** 作该窗口。

构造

HwndSource 的诸多方面只能在构造时指定。 若要创建 HwndSource,请首先创建 HwndSourceParameters 结构,然后使用所需参数填充该结构。 这些参数包括以下内容:

类、窗口和扩展窗口样式。 在创建窗口之后,您必须使用 PInvoke 来更改样式。 并不是所有样式都可以在创建窗口后进行更改。 在更改窗口样式之前,请查阅 Win32 文档。

窗口的初始位置。

窗口的初始大小,其中包括说明是指定此大小,还是应当根据 WPF 内容的已确定大小进行确定。

父窗口。

包含在窗口过程链中的 HwndSourceHook。 如果在构造时指定了挂钩,则它将接收此窗口的所有消息。在创建窗口之后,您可以使用 AddHook 来添加挂钩。

透明度设置。 可以配置顶级窗口,使其根据 WPF 内容的每像素透明度与桌面上的其他窗口混合。 请将HwndSourceParameters 中的 UsesPerPixelOpacity 属性设置为 true 以启用此功能。 此属性只能在构造时通过 HwndSource(HwndSourceParameters) 构造函数签名来指定,并实施多个限制。

在填充 HwndSourceParameters 结构之后,请将其传递到 HwndSource 的HwndSource(HwndSourceParameters) 构造函数。

值得注意的是 对于挂钩 HwndSource 实现的是挂钩将按后进先出的顺序调用 ,所以你需要确定挂钩委托的生命周期

具体代码如下:

//获得HwndSource 有两种方式 用AddHook加委托

//第一种

void win_SourceInitialized(object sender, EventArgs e)

{

System.IntPtr handle = (new System.Windows.Interop.WindowInteropHelper(this)).Handle

WinInterop.HwndSource.FromHwnd(handle).AddHook(new System.Windows.Interop.HwndSourceHook(WindowProc))

}

//第二种

void win_SourceInitialized(object sender, EventArgs e)

{

HwndSource hwndSource = PresentationSource.FromVisual(this) as HwndSource

if (hwndSource != null)

{

hwndSource.AddHook(new HwndSourceHook(WindowProc))

}

}

不知道你的自定义控件是从哪个类派生的,也不知道你是否重载了虚函数WindowProc()一般来说控件自身是没有能力自己处理鼠标消息的,在重载虚函数WindowProc()函数时可以看到其直接调用的是该自定义控件的基类的WindowProc()函数,从而没有搜索当前类的消息映射表,所以我们需要改变消息路由,比如我从Static类派生了一个树形控件,这个控件我需要响应鼠标单击等事件事件这时我需要重载WindowsProc()函数

并添加

if(

message

==

WM_NCHITTEST||

message

==

WM_NCLBUTTONDBLCLK

||message==WM_NCLBUTTONDOWN||message==WM_NCRBUTTONUP)

return

::DefWindowProc(

m_hWnd,

message,

wParam,

lParam

)

return

CStatic::WindowProc(message,

wParam,

lParam)

}

也就是截获WN_NCHITTEST(确定光标所在窗口)消息,鼠标单击消息等;使其搜索该类的消息映射表,从而实现鼠标消息的响应。

新手上路,如有不正确之处请多多包涵。


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/bake/11638326.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-17
下一篇 2023-05-17

发表评论

登录后才能评论

评论列表(0条)

保存