Or, how to safely change a cursor or other resource
In a previous article, I presented a simple class that utilized the IDisposable pattern to set the cursor to a specific value and then automatically reset it to the initial state when some work was accomplished. This simple class had two limitations: it only worked with WPF and it only worked on cursors.
To overcome these limitations, I refactored this to be more general. I changed the context to a generic type and added an abstract RestoreState method that's called when the class is either destroyed (preferred) or the finalizer is called. This allows all the basic plumbing regarding the IDisposable pattern to be contained in one class.
using System;
namespace ResourceProtection
{
public abstract class ResourceChangeHandler : IDisposable
{
protected TContext Context { get; set; }
private bool Disposed { get; set; }
protected ResourceChangeHandler(TContext context)
{
Disposed = false;
Context = context;
}
public void Dispose()
{
DisposeInt();
GC.SuppressFinalize(this);
}
protected abstract void RestoreState(TContext context);
private void DisposeInt()
{
if (Disposed)
return;
RestoreState();
Disposed = true;
}
~ResourceChangeHandler()
{
DisposeInt();
}
}
}
The beauty of this is it can be used as the base class for specific needs. The function of actually changing a resource is split out from the framework needed to manage the change. Here are three examples: one each for a WPF cursor changer, a WinForms cursor changer and a TreeView BeginUpdate/EndUpdate handler.
How to change a WPF cursor
public class WpfChangeCursor : ResourceChangeHandler{
private Cursor OriginalCursor { get; set; }
public WpfChangeCursor(FrameworkElement element, Cursor newCursor)
: base(element)
{
OriginalCursor = Context.Cursor;
Context.Cursor = newCursor;
}
protected override void RestoreState()
{
Context.Cursor = OriginalCursor;
}
}
To use this, instantiate an instance of WpfChangeCursor in a using block, passing in the element who's cursor should be changed and the new value for the cursor. In this case, the context for the base class is the element containing the cursor to be changed. The descendant class has the responsibility to remember the current value and changing the cursor.
How to change a WinForm cursor
public class WinFormChangeCursor : ResourceChangeHandler{
private Cursor OriginalCursor { get; set; }
public WinFormChangeCursor(Cursor newCursor)
: base(Cusor.Current)
{
Cursor.Current = newCursor;
}
protected override void RestoreState()
{
Cursor.Current = Context;
}
}
This class is used by instantiating an instance of WinFormChangeCursor in a using block with the new value of the cursor passed to it. This is a little different from the WPF version since there is only one cursor for the application, rather than for each element. In this case, the context passed to the base is the current cursor value which is then used by RestoreState to put things back.
How to call BeginUpdate and EndUpdate safely
public class TreeViewUpdate : ResourceChangeHandler{
public TreeViewUpdate(TreeView treeview)
: base(treeview)
{
Context.BeginUpdate();
}
protected override void RestoreState()
{
Context.EndUpdate();
}
}
And finally, here's a class to make sure EndUpdate is called for every BeginUpate on a TreeView. Like the WPF cursor changer above, the control is passed in as the context for the ancestor class. BeginUpdate is then called on the context. When the using block this class is controlled by finishes, the EndUpdate in RestoreState is called, again on the same context as the BeginUpdate.
This technique provides for a level of separation between the framework needed to safely do and undo things from the actual work of doing and undoing them. It works well for certain classes of problems, specifically fixed things that happen over and over again. The examples above of doing repetitive things on framework level items are good examples.
Sometimes though, one wants this type of functionality without the overhead of creating a new class each time. In the next article, I'll present a method of composing this type of behavior in a more adhoc manner.
Until then, subscribe to the Twitter feed to get notified of future articles and hear about other development related things. And if you have any questions, suggestions or comments, please feel free to leave them below.
No comments:
Post a Comment