Feb 15
WPF comes with Dependency Properties and everybody using WPF will know about these new kind of properties. When you define your own Dependency Properties in your own class you can pretty easy add a property change event handler:
public int MyProperty
{
get { return (int)GetValue(MyPropertyProperty); }
set { SetValue(MyPropertyProperty, value); }
}
public static readonly DependencyProperty MyPropertyProperty =
DependencyProperty.Register("MyProperty",
typeof(int),
typeof(MainWindow),
new UIPropertyMetadata(0,
MyPropertyvalueChangeCallback));
private static void MyPropertyvalueChangeCallback
(DependencyObject d,
DependencyPropertyChangedEventArgs e)
{
}
But how to add such a event handler to an already existing Dependency Property or somewhere else then in the defining class? E.g. to an property of a WPF-Control that was not build by you. Take a standard WPF-TextBox; both the Text
and the FontSize
properties are Dependency Properties but the TextBox
-class only provides a change event for the Text
-property. Nevertheless you can get a change event for any Dependency Property:
DependencyPropertyDescriptor dpd =
DependencyPropertyDescriptor.FromProperty
(Control.FontSizeProperty, typeof (TextBox));
dpd.AddValueChanged(someTextBox, SomeTextBoxFontSizeChanged);
Every time the FontSizeProperty on the instance someTextBox
changes the given method is called. It’s that easy and you can implement this code everywhere not only within the class that defines the property.
Dec 22
When building applications that mix WPF and Winforms windows (e.g. because you are using a Winforms dialog within a WPF application) you will sometimes need a System.Windows.Forms.IWin32Window interface from an WPF System.Windows.Window. This will be useful when using the WinForms OpenFileDialog and wanting to provide a parent window to the ShowDialog() method that only accepts IWin32Window as parent.
The IWin32Window
interface declares only one read-only property named Handle
that must be provided by the WPF Window.
Implementing the Interface
It is very simple to just add this interface to an existing WPF Window and implementing the single property:
public partial class Window1 : Window, IWin32Window
{
public IntPtr Handle
{
get
{
var interopHelper = new WindowInteropHelper(this);
return interopHelper.Handle;
}
}
}
Using a wrapper class
It does not look as a very good solution to add such an interface to each and every Window
in you whole application only to be able to be the parent of a dialog. So instead of implementing the interface directly you could build a wrapper class that implements this interface and use this wrapper when opening a dialog:
public class Wpf32Window : IWin32Window
{
public IntPtr Handle { get; private set; }
public Wpf32Window(Window wpfWindow)
{
Handle = new WindowInteropHelper(wpfWindow).Handle;
}
}
Opening the dialog will now look like:
d.ShowDialog( new Wpf32Window(this) );
The grand solution to this problem
I like the wrapper solution more than implementing an interface to each and every window but I do not like the changed ShowDialog()
call. It just does not look natural any more. To solve this I created an extension Method to be used with the wrapper class:
public static class CommonDialogExtensions
{
public static DialogResult ShowDialog
(this CommonDialog dialog,
Window parent)
{
return dialog.ShowDialog(new Wpf32Window(parent));
}
}
Now the call to open a WinForms dialog with an WPF parent looks as it should look like:
d.ShowDialog(this);
Dec 22
In fact one OpenFileDialog is enough but the .net framework provides two of them:
- System.Windows.Forms.OpenFileDialog in System.Windows.Forms.dll
- Microsoft.Win32.OpenFileDialog in PresentationFramework.dll (that is WPF)
With such a setup one would use the first for WinForms applications and the second for WPF apps. In fact if you create a WPF application and just type in OpenFileDialog
you will get number 2.
But the 2nd one (even if it is part of the newer part of the framework) is an old dialog where the 1st will use the new and shiny Windows 7 dialog on Win7. Just look at the following screenshots.
Only the first class uses the Windows 7 dialog and comes with proper support for Libraries etc. Even if the second shows the Libraries in its left part it does not fully support this new Windows 7 feature.
I have no idea why the WPF class uses the old windows API but it does. For best compatibility with all Windows versions (including 7) you should always use the Winforms Dialog even when building WPF apps! It does not hurt to reference the System.Windows.Forms.dll (it is installed an every machine running the .net framework) and you can use that dialog from WPF without any trouble.
You should only know about one little disadvantage when using the WinForms dialog. It only accepts WinForms as a parent window no WPF windows. To overcome this limitation please read my other article about the IWin32Window
interface and how to implement / provide it in WPF at http://www.ticklishtechs.net/2009/12/22/how-to-get-a-iwin32window-from-a-wpf-window/
Recent Comments