Click here to Skip to main content
15,886,362 members
Articles / Desktop Programming / WPF

An Error Provider for Windows Presentation Foundation

Rate me:
Please Sign up or sign in to vote.
4.14/5 (13 votes)
19 Jun 20066 min read 84.7K   1.6K   47  
An implementation of an ErrorProvider for Windows Presentation Foundation.
using System;
using System.Data;
using System.Configuration;
using System.Windows;
using System.ComponentModel;
using System.Windows.Controls;
using System.Reflection;
using System.Windows.Data;
using System.Windows.Media;
using System.Collections.Generic;

namespace PaulStovell.Samples.WpfValidation {
    /// <summary>
    /// A Windows Presentaion Foundation error provider.
    /// </summary>
    public class ErrorProvider : FrameworkElement {
        private delegate void FoundBindingCallbackDelegate(FrameworkElement element, Binding binding);
        private FrameworkElement _firstInvalidElement;
        private List<IErrorDisplayStrategy> _displayStrategies;

        /// <summary>
        /// Constructor.
        /// </summary>
        public ErrorProvider() {
            _displayStrategies = new List<IErrorDisplayStrategy>();
            CreateDefaultDisplayStrategies();

            this.DataContextChanged += new DependencyPropertyChangedEventHandler(ErrorProvider_DataContextChanged);
            this.Loaded += new RoutedEventHandler(ErrorProvider_Loaded);
        }

        /// <summary>
        /// Sets up the default inbuilt display strategies.
        /// </summary>
        protected virtual void CreateDefaultDisplayStrategies() {
            AddDisplayStrategy(new TextBoxErrorDisplayStrategy());
        }

        /// <summary>
        /// Called when this component is loaded. We have a call to Validate here that way errors appear from the very 
        /// moment the page or form is visible.
        /// </summary>
        private void ErrorProvider_Loaded(object sender, RoutedEventArgs e) {
            Validate();
        }

        /// <summary>
        /// Adds a display strategy used to display validation errors on a given control.
        /// </summary>
        /// <param name="strategy">The strategy to add.</param>
        public void AddDisplayStrategy(IErrorDisplayStrategy strategy) {
            if (strategy != null && _displayStrategies.Contains(strategy) == false) {
                _displayStrategies.Add(strategy);
            }
        }

        /// <summary>
        /// Removes a display strategy that was added using the AddDisplayStrategy pattern.
        /// </summary>
        /// <param name="strategy">The strategy to remove.</param>
        public void RemoveDisplayStrategy(IErrorDisplayStrategy strategy) {
            if (strategy != null && _displayStrategies.Contains(strategy) == true) {
                _displayStrategies.Remove(strategy);
            }
        }

        /// <summary>
        /// Called when our DataContext changes.
        /// </summary>
        private void ErrorProvider_DataContextChanged(object sender, DependencyPropertyChangedEventArgs e) {
            if (e.OldValue != null && e.OldValue is INotifyPropertyChanged) {
                ((INotifyPropertyChanged)e.NewValue).PropertyChanged -= new PropertyChangedEventHandler(DataContext_PropertyChanged);
            }

            if (e.NewValue != null && e.NewValue is INotifyPropertyChanged) {
                ((INotifyPropertyChanged)e.NewValue).PropertyChanged += new PropertyChangedEventHandler(DataContext_PropertyChanged);
            }

            Validate();
        }

        /// <summary>
        /// Validates all properties on the current data source.
        /// </summary>
        /// <returns>True if there are no errors displayed, otherwise false.</returns>
        /// <remarks>
        /// Note that only errors on properties that are displayed are included. Other errors, such as errors for properties that are not displayed, 
        /// will not be validated by this method.
        /// </remarks>
        public bool Validate() {
            bool isValid = true;
            _firstInvalidElement = null;

            if (this.DataContext is IDataErrorInfo) {
                List<Binding> allKnownBindings = ClearInternal();

                // Now show all errors
                foreach (Binding knownBinding in allKnownBindings) {
                    string errorMessage = ((IDataErrorInfo)this.DataContext)[knownBinding.Path.Path];
                    if (errorMessage != null && errorMessage.Length > 0) {
                        isValid = false;

                        // Display the error on any elements bound to the property
                        FindBindingsRecursively(
                        this.Parent,
                        delegate(FrameworkElement element, Binding binding) {
                            if (knownBinding.Path.Path == binding.Path.Path) {
                                // Figure out which display strategy should be used to display the error, then display it.
                                for (int i = _displayStrategies.Count - 1; i >= 0; i--) {
                                    if (_displayStrategies[i].CanDisplayForElement(element)) {
                                        _displayStrategies[i].DisplayError(element, errorMessage);
                                        if (_firstInvalidElement == null) {
                                            _firstInvalidElement = element;
                                        }
                                        return;
                                    }
                                }
                            }
                        });
                    }
                }
            }
            return isValid;
        }

        /// <summary>
        /// Returns the first element that this error provider has labelled as invalid. This method 
        /// is useful to set the users focus on the first visible error field on a page.
        /// </summary>
        /// <returns></returns>
        public FrameworkElement GetFirstInvalidElement() {
            return _firstInvalidElement;
        }

        /// <summary>
        /// Clears any error messages.
        /// </summary>
        public void Clear() {
            ClearInternal();
        }

        /// <summary>
        /// Clears any error messages and returns a list of all bindings on the current form/page. This is simply so 
        /// it can be reused by the Validate method.
        /// </summary>
        /// <returns>A list of all known bindings.</returns>
        private List<Binding> ClearInternal() {
            // Clear all errors
            List<Binding> bindings = new List<Binding>();
            FindBindingsRecursively(
                    this.Parent,
                    delegate(FrameworkElement element, Binding binding) {
                        // Remember this bound element. We'll use this to display error messages for each property.
                        bindings.Add(binding);

                        // Clear any errors on this framework element.
                        for (int i = _displayStrategies.Count - 1; i >= 0; i--) {
                            if (_displayStrategies[i].CanDisplayForElement(element)) {
                                _displayStrategies[i].ClearError(element);
                            }
                        }
                    });
            return bindings;
        }

        /// <summary>
        /// Called when the PropertyChanged event is raised from the object we are bound to - that is, our data context.
        /// </summary>
        private void DataContext_PropertyChanged(object sender, PropertyChangedEventArgs e) {
            Validate();
        }

        /// <summary>
        /// Recursively goes through the control tree, looking for bindings on the current data context.
        /// </summary>
        /// <param name="element">The root element to start searching at.</param>
        /// <param name="callbackDelegate">A delegate called when a binding if found.</param>
        private void FindBindingsRecursively(DependencyObject element, FoundBindingCallbackDelegate callbackDelegate) {

            // See if we should display the errors on this element
            MemberInfo[] members = element.GetType().GetMembers(BindingFlags.Static | BindingFlags.Public | BindingFlags.GetField | BindingFlags.GetProperty);

            foreach (MemberInfo member in members) {
                DependencyProperty dp = null;

                // Check to see if the field or property we were given is a dependency property
                if (member.MemberType == MemberTypes.Field) {
                    FieldInfo field = (FieldInfo)member;
                    if (typeof(DependencyProperty).IsAssignableFrom(field.FieldType)) {
                        dp = (DependencyProperty)field.GetValue(element);
                    }
                } else if (member.MemberType == MemberTypes.Property) {
                    PropertyInfo prop = (PropertyInfo)member;
                    if (typeof(DependencyProperty).IsAssignableFrom(prop.PropertyType)) {
                        dp = (DependencyProperty)prop.GetValue(element, null);
                    }
                }

                if (dp != null) {
                    // Awesome, we have a dependency property. does it have a binding? If yes, is it bound to the property we're interested in?
                    Binding bb = BindingOperations.GetBinding(element, dp);
                    if (bb != null) {
                        // This element has a DependencyProperty that we know of that is bound to the property we're interested in. 
                        // Now we just tell the callback and the caller will handle it.
                        if (element is FrameworkElement) {
                            if (((FrameworkElement)element).DataContext == this.DataContext) {
                                callbackDelegate((FrameworkElement)element, bb);
                            }
                        }
                    }
                }
            }

            // Now, recurse through any child elements
            if (element is FrameworkElement || element is FrameworkContentElement) {
                foreach (object childElement in LogicalTreeHelper.GetChildren(element)) {
                    if (childElement is DependencyObject) {
                        FindBindingsRecursively((DependencyObject)childElement, callbackDelegate);
                    }
                }
            }
        }
    }
}

By viewing downloads associated with this article you agree to the Terms of Service and the article's licence.

If a file you wish to view isn't highlighted, and is a text file (not binary), please let us know and we'll add colourisation support for it.

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here


Written By
Octopus Deploy
Australia Australia
My name is Paul Stovell. I live in Brisbane and develop an automated release management product, Octopus Deploy. Prior to working on Octopus I worked for an investment bank in London, and for Readify. I also work on a few open source projects. I am a Microsoft MVP for Client Application Development.

Comments and Discussions