WPF主窗体单例模式可以通过以下步骤实现:
在Appxamlcs文件中重写OnStartup方法,在该方法中创建一个MainWindow实例,并将其设置为应用程序的主窗口;
在MainWindowxamlcs文件中重写OnSourceInitialized方法,在该方法中获取当前窗口的句柄,并将其设置为单例模式;
在MainWindowxamlcs文件中重写OnClosed方法,在该方法中释放当前窗口的句柄,以便下次重新创建窗口时可以正常显示;
在Appxamlcs文件中重写OnActivated方法,在该方法中检查当前窗口是否已经存在,如果存在则将其显示出来,如果不存在则创建一个新的窗口。
在学MFC 的第一课就是讲windows的消息机制,其中这一部分是通过WndProc这个函数来完成的,用一个消息循环来捕捉消息。虽然在WPF里 没有 WndProc 这个函数,但是提供了另一种方法来完成挂钩的功能。在WinForm 里 只需重载 WndProc 函数便可完成消息响应。
WinForm:
private const int WM_SIZEING = 0x0214; //窗口改变大小消息
protected override void WndProc(ref SystemWindowsFormsMessage m )
{
switch(mMsg)
{
case WM_SIZEING:
//你的代码
break;
default:
baseWndProc(ref m); // 调用基类函数处理其他消息。
break;
}
}
WPF:
在你运行WPF程序的时候——new App()Run(new MyWindow()),有时候要使用Win32API处理消息的时候该怎么办?其实很简单,一切的需要都在SystemWindowsInterop这个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)
{
SystemIntPtr handle = (new SystemWindowsInteropWindowInteropHelper(this))Handle;
WinInteropHwndSourceFromHwnd(handle)AddHook(new SystemWindowsInteropHwndSourceHook(WindowProc));
}
//第二种
void win_SourceInitialized(object sender, EventArgs e)
{
HwndSource hwndSource = PresentationSourceFromVisual(this) as HwndSource;
if (hwndSource != null)
{
hwndSourceAddHook(new HwndSourceHook(WindowProc));
}
}
在WPF自定义窗体样式时,窗体设置成WindowStyle="None",那么就无法在运行时没有关闭,最小化,最大化按钮。重构窗体时需要加上这三个按钮以及实现功能。
看一下自定义窗体最大化,最小化,关闭效果,鼠标移动动关闭按钮时背景色为红色,表示警告关闭功能,最小化和最大化移动到按钮显示差异颜色。
关闭按钮样式代码如下:Style x:Key="SysCloseButtonStyle" TargetType="{x:Type local:NbToggleButton}" BasedOn="{StaticResource SysButtonStyle}">
<Setter Property="Command" Value="{x:Static shell:SystemCommandsCloseWindowCommand}"/>
<Setter Property="NbData" Value="{StaticResource Icon-Close}"/>
<Setter Property="Template">
<SetterValue>
<ControlTemplate TargetType="{x:Type local:NbToggleButton}">
<Border x:Name="border"
BorderBrush="{TemplateBinding BorderBrush}"
BorderThickness="{TemplateBinding BorderThickness}"
Background="{TemplateBinding Background}"
CornerRadius="{TemplateBinding CornerRadius}"
Effect="{TemplateBinding Effect}"
SnapsToDevicePixels="True">
<Border x:Name="innerBorder"
Background="{TemplateBinding MaskBackground}"
CornerRadius="{TemplateBinding CornerRadius}"
SnapsToDevicePixels="True">
<Viewbox Margin="{TemplateBinding Padding}">
<Path Fill="{TemplateBinding Foreground}"
Data="{TemplateBinding NbData}"
Stretch="Fill"/>
</Viewbox>
</Border>
</Border>
</ControlTemplate>
</SetterValue>
</Setter>
<StyleTriggers>
<Trigger Property="IsMouseOver" Value="True">
<Setter Property="Background" Value="{Binding Path=ColorBrush_Danger1, Source={x:Static local:NbThemeCurrent}}"/>
<Setter Property="MaskBackground" Value="Transparent"/>
<Setter Property="Foreground" Value="{Binding Path=ColorBrush_Text1, Source={x:Static local:NbThemeCurrent}}"/>
</Trigger>
<Trigger Property="IsPressed" Value="True">
<Setter Property="Background" Value="{Binding Path=ColorBrush_Danger1, Source={x:Static local:NbThemeCurrent}}"/>
<Setter Property="MaskBackground" Value="{Binding Path=ColorBrush_Base2_OP1, Source={x:Static local:NbThemeCurrent}}"/>
<Setter Property="Foreground" Value="{Binding Path=ColorBrush_Text1, Source={x:Static local:NbThemeCurrent}}"/>
</Trigger>
</StyleTriggers>
</Style>
你好,很高兴你的问题描述思路清晰具体
对于你提到的media foundation,我们项目中使用的是wpf内嵌的mediaelement,足以应对大部分的需求对于media foundation以前尝试研究过,但是效果不大理想,因为对于很多视频要求的控制不能很好的耦合,自己处理比较麻烦
如果有必须的要求,我现在搜集以前的一些资料,请稍微等待一下
--------------------
hi,我们同事说,media foundation 已经合并到了 Windows Media Player 中,请参考如下文章进行调试
系统bug,网络问题。
1、系统bug是wpfspy++系统出现了问题导致无法获取窗口,等待官方修复即可。
2、网络问题是自身设备连接的网络出现较大波动,导致wpfspy++无法获取窗口,更换网络重新打开即可。
应该是你调用顺序没对,一般可以考虑在某个按钮点击里或者pageload里调用。
而你的代码看上去是window(或者gdShow)都还没创建完毕,就调用了Display。
试试把Display方法移动到最后?
私信帮改
需要在WPF中通过XAML将窗口的DataContext指定为this很简单,如下所示:
关键即在倒数第5和6行:
达叔傻乐( darwinzuo@163com )
以上就是关于WPF主窗体单例模式要求主程序不能开启多个全部的内容,包括:WPF主窗体单例模式要求主程序不能开启多个、wpf usercontrol窗口怎么使用windowproc、WPF编程怎么激活指定窗口等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)