Click here to Skip to main content
65,938 articles
CodeProject is changing. Read more.

Hourglass Mouse Cursor Always Changes Back to its Original Image. How?

4.85/5 (11 votes)
21 Dec 2010CPOL1 min read 29.5K  
IDisposable with “using” block helps in all cases
Testing and debugging UI code, I am often faced with such a situation, when the mouse cursor is changed to Hourglass, but not changed back.
Sometimes, it happened to my own code as well.

The mouse cursor is supposed to be changed back to its original or default image after the procedure which could cause a delay if UI response is already completed (or failed). Sometimes it is not changed back because this procedure fails due to exception or some other reason. Sometimes the code used to set mouse cursor to its original image is simply forgotten.

As a result, I came to a simple pattern which helps to avoid such situation in all cases. I used it in several different UI systems. Now I'll show how it is implemented with .NET. The idea is using helper class implementing IDisposable.

This is a WPF implementation written in C#:

C#
namespace SA.Universal.UI {
    using IDisposable = System.IDisposable;
    using FrameworkElement = System.Windows.FrameworkElement;
    using Cursor = System.Windows.Input.Cursor;
    using Cursors = System.Windows.Input.Cursors;
    using Debug = System.Diagnostics.Debug;

    public class WaitCursorIndicator : IDisposable {

        public WaitCursorIndicator(FrameworkElement owner) {
            this.Onwer = owner;
            Debug.Assert(
                owner != null,
                "WaitCursorIndicator expects non-null argument");
            if (owner == null) return;
            Previous = owner.Cursor;
            owner.Cursor = Cursors.Wait;
        } //WaitCursorIndicator

        void IDisposable.Dispose() {
            if (this.Onwer == null) return;
            this.Onwer.Cursor = Previous;
        } //IDisposable.Dispose

        FrameworkElement Onwer;
        Cursor Previous;

    } //class WaitCursorIndicator

} //namespace SA.Universal.UI


With System.Windows.Forms, it is implemented in a very similar way:

C#
namespace SA.Universal.UI {
    using System.Windows.Forms;
    using IDisposable = System.IDisposable;
    using Debug = System.Diagnostics.Debug;

    public class WaitCursorIndicator : System.IDisposable {

        public WaitCursorIndicator(Control owner) {
            this.Owner = owner;
            Debug.Assert(
                owner != null,
                "WaitCursorIndicator expects non-null argument");
            if (owner == null) return;
            Previous = this.Owner.Cursor;
            owner.Cursor = Cursors.WaitCursor;
        } //WaitCursorIndicator

        void IDisposable.Dispose() {
            if (Owner == null) return;
            Owner.Cursor = Previous;
        } //IDisposable.Dispose

        Control Owner;
        Cursor Previous;

    } //class WaitCursorIndicator

} //namespace SA.Universal.UI


The usage looks identical for WPF and System.Windows.Forms:

C#
using(new WaitCursorIndicator(owner)) {
   //... some long-running code here
} //end using


In this code snippet, owner is most typically a form or a window.

This way, mouse cursor is changed back no matter what, even if an exception is thrown.
On exit from the "using" block, WaitCursorIndicator.Dispose is always called. In fact, using this "using" block is just syntactic sugar functionally strictly equivalent to the try-finally block with Dispose called in the finally section.

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)