WPF工作笔记:本地化支持、主进程通知、两种最常用异步编程方式
1、本地化支持
(1)重写控件默认的依赖属性LanguageProperty
FrameworkElement.LanguageProperty.OverrideMetadata( typeof(FrameworkElement), new FrameworkPropertyMetadata( XmlLanguage.GetLanguage(CultureInfo.CurrentCulture.IetfLanguageTag)));
(2)在项目资源文件中添加Resources.resx,Resources.zh-CN.resx等等本地化字符串资源。
文件中依次输入Key-Value键值对,key相同,Value中输入本地化语言字符串,如下所示
(3)在界面中引用资源文件所在命名空间。如:
xmlns:resc="clr-namespace:WpfApplication.Resources"
(4)引用字符串资源。如 :
<Button Content="{x:Static resc:Resources.Open}"/>
2、主进程通知
在多文件支持和音视频播放界面应用中,经常需要在外部通知主进程。比如您已经打开了酷狗音乐播放界面,在外部文件夹中,双击.mp3音乐文件,不会启动第二个酷狗音乐播放界面,而是通知主窗口接收外部请求。下面给出一种实现方式:
首先声明WPF开发的系统只有一个入口,那就是Application类,也就是App.xaml
(1)App.xaml中不需要StartupUri,在后台启动主窗口。首先在App.xaml.cs中定义事件句柄,用于通知主窗口
public static EventWaitHandle ProgramStarted; private bool CreatedNew;
(2)App.xaml.cs重写OnStartup,判断是否需要新建主窗口
ProgramStarted = new EventWaitHandle(false, EventResetMode.AutoReset, "MyStartEvent", out CreatedNew);
通过上面语句得到的CreatedNew值来决定是否需要创建主窗口,CreatedNew等于true时按一般情况声明主窗口:
if (CreatedNew) { MainWindow mainWin = new MainWindow(); mainWin.Show(); }
(3)下面才是重点,在主窗口已经启动的情况下,如何将外部的文件传递到主窗口去
因为前面已经定义了事件句柄ProgramStarted,我们首先需要将外部文件写入到一个本地文件或者注册表中(名其为Global),已经启动的窗口进程得到通知后,去读取新建App进程写入的内容。
然后调用ProgramStarted.Set();Thread.Sleep(100);主窗口就可以得到通知。
(4)主窗口的线程池中需要注册Wait请求:
MainWindow.xaml.cs构造函数中:
ThreadPool.RegisterWaitForSingleObject(App.ProgramStarted, OnProgramStarted, null, -1, false);
事件回调处理函数:
/// <summary> /// 主进程在接收到其他进程通知后回调函数 /// </summary> /// <param name="state"></param> /// <param name="timeout"></param> private void OnProgramStarted(object state, bool timeout) { Thread thread = new Thread(new ThreadStart(new Action(() => { this.Dispatcher.BeginInvoke(System.Windows.Threading.DispatcherPriority.Normal, (System.Threading.ThreadStart)delegate() { //处理Global }); }))); //因为是线程池通知主进程,必须在单线程单元ApartmentState.STA执行 thread.SetApartmentState(ApartmentState.STA); thread.IsBackground = true; thread.Start(); }
注意上面代码中使用了Dispatcher,只有WPF这样使用,能够访问主窗口控件,因为主窗口控件所在的线程是主窗口,不能在新建的线程中调用,如果要使用,需要使用Dispatcher。
默认Dispatcher执行的环境也是ApartmentState.STA,所以上面的代码可以简单为(只能在WPF中可以这样使用):
private void OnProgramStarted(object state, bool timeout) { this.Dispatcher.BeginInvoke(System.Windows.Threading.DispatcherPriority.Normal, (System.Threading.ThreadStart)delegate() { //处理Global }); }
3、两种最常用异步编程
(1)DispatcherObject
this.Dispatcher.BeginInvoke(DispatcherPriority.Normal, (ThreadStart)delegate() { // Simulate some work taking place. Thread.Sleep(TimeSpan.FromSeconds(5)); btnContent.Text = "Here is some new text."; } );
(2)BackgroundWorker
BackgroundWorker是WPF异步编程经常使用的类,使用方式比较正规,很容易学会。
BackgroundWorker bgw = new BackgroundWorker(); bgw.WorkerSupportsCancellation = true; bgw.WorkerReportsProgress = true; bgw.DoWork += new DoWorkEventHandler(bgw_DoWork); bgw.RunWorkerCompleted += new RunWorkerCompletedEventHandler(bgw_RunWorkerCompleted); bgw.RunWorkerAsync(argument);
//bgw_DoWork不能直接使用主线程元素,只能通过e.Argument得到主线程传过来的参数 private void bgw_DoWork(object sender, DoWorkEventArgs e) { string argument= e.Argument.ToString(); e.Result = new LongTimeFun(argument); } private void bgw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e) { MyObject obj = e.Result as MyObject ; //这个时候才能使用主界面控件元素 } private MyObject LongTimeFun(string argument){ }
简单形式(Lambda语法):
BackgroundWorker bgw = new BackgroundWorker(); bgw.WorkerSupportsCancellation = true; bgw.WorkerReportsProgress = true; bgw.DoWork += new DoWorkEventHandler((sender, e) => { e.Result = new LongTimeFun(argument); }); bgw.RunWorkerCompleted += new RunWorkerCompletedEventHandler((sender, e) => { MyObject obj = e.Result as MyObject ; });
(3)DispatcherObject和BackgroundWorker结合使用
我们知道BackgroundWorker的DoWork事件回调事件中不能使用界面元素,但是在WPF中可以结合使用DispatcherObject、BackgroundWorker达到目的,
这样就不需要BackgroundWorker的RunWorkerCompleted回调事件了。(这个几乎在书本和其它技术资料中没有介绍,但是很多时候是很有用的)
BackgroundWorker bgw = new BackgroundWorker(); bgw.WorkerSupportsCancellation = true; bgw.WorkerReportsProgress = true; bgw.DoWork += new DoWorkEventHandler((sender, e) => { this.Dispatcher.BeginInvoke(System.Windows.Threading.DispatcherPriority.Normal, (System.Threading.ThreadStart)delegate() { //这样DoWork回调事件中可以使用主界面控件元素 }); }); bgw.RunWorkerAsync();
4、滚动条内容设置可见
(1)使用FrameworkElement方法 BringIntoView
FrameworkElement.BringIntoView();
(2)ListBox
listbox.ScrollIntoView(listbox.Items[index]);
(3)Scrollviewer
scrollViewer.ScrollToVerticalOffset(VisualTreeHelper.GetOffset(VisualObject).Y);