文章出處

由于Binding只把Convert成功的值送往Source,當目標中的值Convert失敗時Source的值依然是舊值,所以ViewModel必須獲取View的輸入驗證狀態,以下是本人的實現。

當“+”號兩邊輸入正確時,“Add”可用,當所有“+”號兩邊輸入正確時,“Add All”可用。

通過Behavior添加Validation.ErrorEvent路由事件的事件處理器,在該事件處理器中把HasError狀態寫入自定義的附加屬性,附加屬性可以綁定。

Behavior派生類引用System.Windows.Interactivity.dll,代碼如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Interactivity;

namespace Calculater
{
    public class NotifyErrorBehavior : Behavior<UIElement>
    {
        protected override void OnAttached()
        {
            base.OnAttached();

            this.AssociatedObject.AddHandler(Validation.ErrorEvent, new RoutedEventHandler(HasErrorChanged));
        }

        protected override void OnDetaching()
        {
            base.OnDetaching();
            this.AssociatedObject.RemoveHandler(Validation.ErrorEvent, new RoutedEventHandler(HasErrorChanged));
        }

        private static void HasErrorChanged(object sender, RoutedEventArgs e)
        {
            DependencyObject d = e.OriginalSource as DependencyObject;
            HasErrorHelper.SetHasError(d, Validation.GetHasError(d));
        }
    }
}

附加屬性代碼:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;

namespace Calculater
{
    public class HasErrorHelper
    {
        public static bool GetHasError(DependencyObject obj)
        {
            return (bool)obj.GetValue(HasErrorProperty);
        }

        public static void SetHasError(DependencyObject obj, bool value)
        {
            obj.SetValue(HasErrorProperty, value);
        }

        public static readonly DependencyProperty HasErrorProperty =
            DependencyProperty.RegisterAttached("HasError", typeof(bool), typeof(HasErrorHelper), new PropertyMetadata(false));
    }
}

View代碼:

<UserControl x:Class="Calculater.ChildCalculater"
             xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
             xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
             xmlns:local="clr-namespace:Calculater"
             xmlns:i="http://schemas.microsoft.com/expression/2010/interactivity"
             mc:Ignorable="d" 
             d:DesignHeight="80" d:DesignWidth="500">
    <i:Interaction.Behaviors>
        <local:NotifyErrorBehavior></local:NotifyErrorBehavior>
    </i:Interaction.Behaviors>
    <Grid>
        <Grid.ColumnDefinitions>
            <ColumnDefinition />
            <ColumnDefinition Width="Auto"/>
            <ColumnDefinition />
            <ColumnDefinition Width="Auto"/>
            <ColumnDefinition />
            <ColumnDefinition Width="Auto"/>
            <ColumnDefinition Width="Auto"/>
        </Grid.ColumnDefinitions>
        <TextBox Margin="3" HorizontalContentAlignment="Center" VerticalContentAlignment="Center" local:HasErrorHelper.HasError="{Binding Path=HasErrorX,Mode=OneWayToSource}">
            <Binding Path="X" NotifyOnValidationError="True" UpdateSourceTrigger="PropertyChanged" TargetNullValue="" />
        </TextBox>
        <Label Margin="3" VerticalContentAlignment="Center" HorizontalContentAlignment="Center" Grid.Column="1">+</Label>
        <TextBox Margin="3" HorizontalContentAlignment="Center" VerticalContentAlignment="Center" Grid.Column="2" local:HasErrorHelper.HasError="{Binding Path=HasErrorY,Mode=OneWayToSource}">
            <Binding Path="Y" NotifyOnValidationError="True" UpdateSourceTrigger="PropertyChanged" TargetNullValue="" />
        </TextBox>
        <Label Margin="3" VerticalContentAlignment="Center" HorizontalContentAlignment="Center" Grid.Column="3">=</Label>
        <TextBox Margin="3" HorizontalContentAlignment="Center" VerticalContentAlignment="Center" Grid.Column="4" IsReadOnly="True">
            <Binding Path="Sum" TargetNullValue="" />
        </TextBox>
        <Button Margin="3" Grid.Column="5" Padding="3" Command="{Binding CalculateCommand}">Add</Button>
        <Button Margin="3" Grid.Column="6" Padding="3" Command="{Binding ResetCommand}">Reset</Button>
    </Grid>
</UserControl>

ViewModel引用Microsoft.Practices.Prism.dll,代碼如下:

using Microsoft.Practices.Prism.Commands;
using Microsoft.Practices.Prism.ViewModel;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

namespace Calculater
{
    class ChildCalculaterViewModel : NotificationObject
    {
        public ChildCalculaterViewModel()
        {
            this._calculateCommand = new DelegateCommand(this.Calculate, this.CanCalculate);
            this._resetCommand = new DelegateCommand(this.Reset);

            CalculaterCommand.CalculateAllCommand.RegisterCommand(this.CalculateCommand);
            CalculaterCommand.ResetAllCommand.RegisterCommand(this.ResetCommand);
            
        }

        private double? _x;

        public double? X
        {
            get { return _x; }
            set
            {
                if (_x != value)
                {
                    _x = value;
                    this.RaisePropertyChanged("X");
                    this.Sum = null;
                    this.CalculateCommand.RaiseCanExecuteChanged();
                }

            }
        }

        private double? _y;

        public double? Y
        {
            get { return _y; }
            set
            {
                if (_y != value)
                {
                    _y = value;
                    this.RaisePropertyChanged("Y");
                    this.Sum = null;
                    this.CalculateCommand.RaiseCanExecuteChanged();
                }

            }
        }

        private double? _sum;

        public double? Sum
        {
            get { return _sum; }
            set
            {
                if (_sum != value)
                {
                    _sum = value;
                    this.RaisePropertyChanged("Sum");
                }

            }
        }

        private bool _hasErrorX;

        public bool HasErrorX
        {
            get { return _hasErrorX; }
            set
            {
                if (_hasErrorX != value)
                {
                    _hasErrorX = value;
                    this.Sum = null;
                    this.CalculateCommand.RaiseCanExecuteChanged();
                }

            }
        }

        private bool _hasErrorY;

        public bool HasErrorY
        {
            get { return _hasErrorY; }
            set
            {
                if (_hasErrorY != value)
                {
                    _hasErrorY = value;
                    this.Sum = null;
                    this.CalculateCommand.RaiseCanExecuteChanged();
                }

            }
        }

        private DelegateCommand _calculateCommand;

        public DelegateCommand CalculateCommand
        {
            get { return _calculateCommand; }
        }

        private bool CanCalculate()
        {
            if (this.HasErrorX || this.HasErrorY || !this.X.HasValue || !this.Y.HasValue)
            {
                return false;
            }

            return true;
        }

        private void Calculate()
        {
            try
            {
                double x = this.X.Value;
                double y = this.Y.Value;

                this.Sum = x + y;
            }
            catch
            {
                return;
            }
        }

        private DelegateCommand _resetCommand;

        public DelegateCommand ResetCommand
        {
            get { return _resetCommand; }
        }

        private void Reset()
        {
            this.X = null;
            this.Y = null;
            this.Sum = null;
        }
    }
}

主窗體代碼:

<Window x:Class="Calculater.Shell"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:local="clr-namespace:Calculater"
        xmlns:prism="http://www.codeplex.com/prism"
        Title="Calculater" Height="300" Width="600">
    <Grid>
        <Grid.RowDefinitions>
            <RowDefinition Height="Auto"/>
            <RowDefinition Height="Auto"/>
            <RowDefinition Height="Auto"/>
            <RowDefinition Height="Auto"/>
            <RowDefinition Height="Auto"/>
            <RowDefinition />
        </Grid.RowDefinitions>
        <local:ChildCalculater></local:ChildCalculater>
        <local:ChildCalculater Grid.Row="1"></local:ChildCalculater>
        <local:ChildCalculater Grid.Row="2"></local:ChildCalculater>
        <Grid Grid.Row="3">
            <Grid.ColumnDefinitions>
                <ColumnDefinition />
                <ColumnDefinition />
            </Grid.ColumnDefinitions>
            <Button Margin="3" HorizontalAlignment="Right" Padding="3" Content="Add All" Command="{x:Static local:CalculaterCommand.CalculateAllCommand}"></Button>
            <Button Margin="3" HorizontalAlignment="Left" Padding="3" Grid.Column="1" Content="Reset All" Command="{x:Static local:CalculaterCommand.ResetAllCommand}"></Button>
        </Grid>
    </Grid>
</Window>
using Microsoft.Practices.Prism.Commands;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace Calculater
{
    public static class CalculaterCommand
    {
        private static CompositeCommand _calculateAllCommand = new CompositeCommand();

        public static CompositeCommand CalculateAllCommand
        {
            get { return _calculateAllCommand; }
        }

        private static CompositeCommand _resetAllCommand = new CompositeCommand();

        public static CompositeCommand ResetAllCommand
        {
            get { return _resetAllCommand; }
        }

    }
}

 


文章列表




Avast logo

Avast 防毒軟體已檢查此封電子郵件的病毒。
www.avast.com


arrow
arrow
    全站熱搜
    創作者介紹
    創作者 大師兄 的頭像
    大師兄

    IT工程師數位筆記本

    大師兄 發表在 痞客邦 留言(0) 人氣()