【WPF学习】第二十七章 Application类的任务
上一章介绍了有关WPF应用程序中使用Application对象的方式,接下来看一下如何使用Application对象来处理一些更普通的情况,接下俩介绍如何初始化界面、如何处理命名行参数、如何处理支付窗口之间的交互、如何添加跟踪文档以及如何创建单示例应用程序。
一、显示初始化界面
WPF应用程序的运行速度快,但并不能在瞬间启动。当第一次启动应用程序时,会有一些延迟,因为公共语言运行时(Common Language Runtime,CLR)首先需要初始化.NET环境,然后启动应用程序。
这一延迟未必成为问题。通常,只需要经历很短的时间,就会出现第一个窗口,但如果具有更耗时的初始化步骤,或者如果只是希望通过显示打开的图像应用程序显得更加专业,这时可使用WPF提供的简单初始界面特性。
下面是添加初始界面的方法:
(1)为项目添加图像文件(通常是.bmp、.png或.jpg文件)。
(2)在Solution Explorer中选择图像文件。
(3)将Build Action修改为SplashScreen。
下次运行应用程序时,图像会立即在屏幕中央显示出来。一旦准备好运行时环境。而且Application_Startup方法执行完毕,应用程序的第一个窗口就将显示出来,这时初始化界面图像会很快消失(约需300毫秒).
该特性听起来很简单,事实上也确实如此。只需要记住显示的初始化界面没有任何装饰,在它周围没有窗口边框,所有由你决定是否为初始界面图像添加边框,也无法通过显示一系列的多幅图像或动画让初始化图像显得更富有想象力的效果。如果希望得到这种效果,需要采用传统方法:创建在运行初始化代码的同事显示你所希望的图像界面的启动窗口。
顺便提一下,当添加初始界面时,WPF编译器会自动生成的App.g.cs文件添加与下面的类似的代码:
SplashScreen splashScreen = new SplashScreen("images/blue.jpg");
//Show the splash screen
//The true parameter sets the splashScreen to fade away automatically
//after the first window appears
splashScreen.Show(true); //Start the application
AssemblyResources.App app = new AssemblyResources.App();
app.InitializeComponent();
app.Run();
//The splash screen begins ites automatic fade-out now.
也可自行编写这一剪短逻辑,而不死使用SplashScreen 生成操作。但有一点需要指出,可以改变的唯一细节是初始界面褪去的速度。为此,需要项SplashScreen.Show()方法传递false(从而使WPF不会自动淡入初始化界面)。然后由你负责通过调用SplashScreen.Close()方法在恰当的时机隐藏初始界面,并提供TimeSpan值来指示经过多长时间淡出初始化界面。
二、处理命令行参数
为处理命令行参数,需要相应Application.Startup事件。命令行参数是通过StartupEventArgs.Args属性作为字符串数组提供的。
例如,假定希望加载文档,文档名作为命令行参数传递。在这种情况下,有必要读取命令行参数并执行所需的一些额外初始化操作。在下面的示例中,通过响应Application.Startup事件实现了这一模式。在该例中,没有在任何地方设置Application.StartupUri属性——而是使用代码实例化窗口。
public partial class App : Application
{
// The command-line argument is set through the Visual Studio
// project properties (the Debug tab).
private void App_Startup(object sender, StartupEventArgs e)
{
// At this point, the main window has been created but not shown.
FileViewer win = new FileViewer(); if (e.Args.Length > )
{
string file = e.Args[];
if (File.Exists(file))
{
// Configure the main window.
win.LoadFile(file);
}
} // This window will automatically be set as the Application.MainWindow.
win.Show();
}
}
<Application x:Class="LoadFromCommandLine.App"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Startup="App_Startup">
<Application.Resources> </Application.Resources>
</Application>
App.xaml
上面的方法初始化主窗口,然后当App_Startup()方法结束是显示主窗口。上面的代码假定FileViewer类有名为LoadFile()的共有方法。这只是一个示例,它只读取并显示指定文件的文本。
三、访问当前Application对象
通过静态的Application.Current属性,可在应用程序的任何位置获取当前应用程序实例,从而在窗口之间进行基本交互,因为任何窗口都可以访问当前Application对象,并通过Application对象,并通过Application对象获取主窗口的作用:
Window main=Application.Current.MainWindow;
MessageBox.Show("The main window is "+main.Title);
当然,如果希望访问在自定义主窗口类中添加的任意方法、属性或事件,需要将窗口对象转换为正确类型。如果主窗口的自定义类MainWindow的实例,可使用与下面类似的代码:
MainWindow main=(MainWindow)Application.Current.MainWindow;
main.DoSomething();
在窗口中还可以检查Application.Windows集合的内容,该属性提供了所有当前打开窗口的引用:
foreach(Window window in Application.Current.Windows)
{
MessageBox.Show(window.Title+" is open.");
}
实际上,大多数应用程序通常使用一种更具结构化特点的方式在窗口之间进行交互。如果有几个长时间运行的窗口同事打开,并且它们之间需要以某种方式进行通信,在自定义应用程序类中保存这些窗口的引用可能更有意义。这样,总可以找到所需的窗口。与此类似,如果有基于文档的应用程序,那么可选择创建跟踪文档窗口的集合,而不是跟踪其他内容。
四、在窗口之间进行交互
整如在前面已经看到的,自定义应用程序类是放置响应不同应用程序事件的代码的好地方。应用程序类还可以很好地达到另一个目的:保存重要窗口的引用,使一个窗口可访问另一个窗口。
例如,假设希望跟踪应用程序使用的所有文档窗口。为此,可在自定义应用程序类中创建专门的集合。下面是使用泛型列表集合保存一组自定义窗口对象的示例。在这个示例中,每个文档窗口由名为Document类的实例表示:
public partial class App : Application
{
private List<Document> documents = new List<Document>(); public List<Document> Documents
{
get { return documents; }
set { documents = value; }
}
}
现在,当创建新文档时,只需要记住将其添加到Documents集合中即可。下面是响应按钮单击事件的事件处理程序,该事件处理程序完成了所需的工作:
private void cmdCreate_Click(object sender, RoutedEventArgs e)
{
Document doc = new Document();
doc.Owner = this;
doc.Show();
((App)Application.Current).Documents.Add(doc);
}
同样,也可在Document类中响应Window.Loaded类这些事件,以确保当创建文档对象时,总会在Documents集合中注册该文档对象。
现在,可在代码的其他任何地方进行集合来遍历所有文档,并使用公有成员。在该例中,Document类包含用于更新显示的自定义方法SetContent();
private void cmdUpdate_Click(object sender, RoutedEventArgs e)
{
foreach (Document doc in ((App)Application.Current).Documents)
{
doc.SetContent("Refreshed at " + DateTime.Now.ToLongTimeString() + ".");
}
}
<Window x:Class="WindowTracker.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="WindowTracker" Height="300" Width="300"
>
<StackPanel>
<Button Click="cmdCreate_Click" Margin="10,10,10,5" Name="cmdCreate">Click to Create a Document</Button>
<Button Click="cmdUpdate_Click" Margin="10,5,10,10" Name="cmdUpdate">Click to Refresh the Documents</Button>
</StackPanel>
</Window>
MainWindow.xaml
<Window x:Class="WindowTracker.Document"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="WindowTracker" Height="78" Width="209"
>
A new document.
</Window>
Document.xaml
public partial class Document : Window
{
public Document()
{
InitializeComponent();
} public void SetContent(string content)
{
this.Content = content;
}
}
Document.xaml.cs
下图显示了最终效果图。最终结果谈不上华美,但这种交互方式值得注意——演示了一种通过自定义应用程序类在窗口之间进行交互的安全规范的方式。这种方式比使用Window属性要好,因为是强类型,只包含Document窗口(而不是包含窗口应用程序中所有窗口的集合)。通过这种方式还可使用另一种更有用的方式对窗口进行分类。例如,可使用字典集合,通过键名更方便地查找文档。在基于文档的引用程序中,可通过文件名来索引集合中的窗口。
五、单实例应用程序
通常,只要愿意就可以加载WPF应用程序的任意多个副本。某些情况下, 这种设计时非常合理的。但在另外一些情况下,这可能会成为问题,当构建基于文档的应用程序时更是如此。
对于单实例应用程序,WPF本身并未提供自带的解决方法,但可使用几种变通方法。基本技术时当触发Application.Startup事件时,检查另一应用程序实例是否已在运行。最简单的方法是使用全局的mutex对象(mutex对象时操作系统提供的用于进程间通信的同步对象)。这种方法很简单,但功能有限。最重要的是,应用程序的新实例无法与已经存在的实例进行通信。对于基于文档的应用程序而言这确实是一个问题,因为新实例可能需要告诉已经存在的应用程序实例打开某个特定的文档(如果该文档是通过命令行参数传递的)。
【WPF学习】第二十七章 Application类的任务的更多相关文章
- 《Linux命令行与shell脚本编程大全》 第二十七章 学习笔记
第二十七章:shell脚本编程进阶 监测系统统计数据 系统快照报告 1.运行时间 uptime命令会提供以下基本信息: 当前时间 系统运行的天数,小时数,分钟数 当前登录到系统的用户数 1分钟,5分钟 ...
- Gradle 1.12用户指南翻译——第二十七章. Ear 插件
其他章节的翻译请参见: http://blog.csdn.net/column/details/gradle-translation.html 翻译项目请关注Github上的地址: https://g ...
- “全栈2019”Java多线程第二十七章:Lock获取lock/释放unlock锁
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java多 ...
- “全栈2019”Java第八十七章:类中嵌套接口的应用场景(拔高题)
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- “全栈2019”Java第三十七章:类与字段
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- “全栈2019”Java第二十七章:流程控制语句中循环语句for
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- 风炫安全WEB安全学习第二十七节课 XSS的防御措施
风炫安全WEB安全学习第二十七节课 XSS的防御措施 XSS防御措施 总的原则 控制好输入/输出 过滤:根据业务需求进行过滤,对email,手机号码这样的输入框进行验证. 转义:所有输出到前端的数据都 ...
- 【WPF学习】第二十六章 Application类——应用程序的生命周期
在WPF中,应用程序会经历简单的生命周期.在应用程序启动后,将立即创建应用程序对象,在应用程序运行时触发各种应用程序事件,你可以选择监视其中的某些事件.最后,当释放应用程序对象时,应用程序将结束. 一 ...
- Java基础知识二次学习--第六章 常用类
第六章 常用类 时间:2017年4月26日16:14:49~2017年4月26日16:56:02 章节:06章_01节~06章_06节 视频长度:20:57+1:15+8:44+1:26+11:2 ...
随机推荐
- markdown设置编辑基本语法
看到其他人写的东西,版面设计,文字样式,区域划分都是那么好看,我一直不知道是怎么设计的,今天发现了,做以记录. #一.设置Markdown编辑模式 二.Markdown编辑语法 一.标题 在想要设置为 ...
- 019 Ceph整合openstack
一.整合 glance ceph 1.1 查看servverb关于openstack的用户 [root@serverb ~]# vi ./keystonerc_admin unset OS_SERVI ...
- echarts在一个折线/柱状图浮窗显示多条数据
解决问题就在data里面,首先 data里面是可以json数组形式,如官方API上的 name:”“, value:”“,等 value是echart识别折线图的key值.1.来看数据格式 data: ...
- 「Luogu P1210」回文检测 解题报告
题面 这是一道诡异的黄题 居然让你求一串吧啦吧啦的东西中 字母(大小写)最长的回文串的长度,还要输出完整的串 吐血 思路: 保持淡定,我们啥都不会,就会Manacher,那就用Manacher大法! ...
- 学了java,我才发现台球还可以这样玩!
桌球小游戏的尝试 桌球是人们日常生活中都能接触到的一种娱乐活动,随着互联网技术的发展,手机上也有了很多桌球小游戏,让人们随时随地都能打两把. 今天分享一个用java编写的桌球小游戏 代码如下: ...
- samba服务器红帽5.4搭建,亲测可用!!!
samba服务器搭建 服务器的环境 红帽5.4 vm15 挂载光盘 mount mount -t iso9660 设备目录 /mnt 表示挂载 软件包安装 samba服务器只需安装两个软件包,先找到软 ...
- 06_URL参数截取
1:如何获取URL传给子页面的参数: //获得参数(只对字母数字等有效,参数值为中文则不能传) function getQueryString(name) { var reg = new RegExp ...
- Spring Cloud Stream消息驱动@SendTo和消息降级
参考程序员DD大佬的文章,自己新建demo学习学习,由于需要消息回执,看到了@SendTo这个注解能够实现,下面开始学习demo,新建两个项目cloud-stream-consumer消费端 和 cl ...
- Linux中的零拷贝
零拷贝 本文图片和一些内容均来自后面的参考,非原创只是把文章中的一些关键内容整理一下,算作是一个学习笔记. 传统的I/O操作 传统的IO操作是用户应用程序只是需要调用两个系统调用 read() 和 w ...
- ACM北大暑期课培训第五天
今天讲的扫描线,树状数组,并查集还有前缀树. 扫描线 扫描线的思路:使用一条垂直于X轴的直线,从左到右来扫描这个图形,明显,只有在碰到矩形的左边界或者右边界的时候,这个线段所扫描到的情况才会改变, ...