1

我是 WPF 开发的新手。我只是快速创建了一个示例来重新创建场景。我正在根据父行中的复选框在 rowdetailsview 中切换数据网格的 IsReadonly 属性。

除了在一种特定情况下,一切正常。

如何重现问题。

保持选中最初创建的父行。取消选中父行。转到子行 id 属性。清除 id 字段中的所有内容并从该单元格中取出标签,您将看到空引用异常。

我不知道如何解决这个问题。任何见解都会非常有用。

代码后面的代码:

 namespace WpfApplication7
{/// <summary>
/// Interaction logic for MainWindow.xaml
/// </summary>
public partial class MainWindow : Window
{

    public PerColl People { get; private set; }
    public MainWindow()
    {
        InitializeComponent();

        this.People = new PerColl();
        this.DataContext = this;
    }

    private void Button_Click(object sender, RoutedEventArgs e)
    {

    }

    public class Person
   : INotifyPropertyChanged, IEditableObject
    {
        public string Name { get; set; }

        public double Salary
        {
            get { return _salary; }
            set
            {
                if (this._salary == value)
                    return;
                this._salary = value;
                if (this.PropertyChanged != null)
                    this.PropertyChanged(this, new PropertyChangedEventArgs("Salary"));
            }

        }

        public bool IsLocked
        {
            get { return _isLocked; }
            set
            {

                if (this._isLocked == value)
                    return;
                this._isLocked = value;
                if (this.PropertyChanged != null)
                    this.PropertyChanged(this, new PropertyChangedEventArgs("IsLocked"));
            }
        }

        public ObservableCollection<Kid> Kids { get; set; }

        public Person(string name, double salary)
        {
            this.Name = name;
            this.Salary = salary;
        }


        public Person()
        {
            this.Salary = 10000;
            this.Name = "abc";
            this.IsLocked = true;
            this.Kids = new ObservableCollection<Kid>();
            this.Kids.Add(new Kid(1));
            this.Kids.Add(new Kid(2));
        }

        private bool _isLocked;
        private double _salary;
        #region INotifyPropertyChanged Members

        public event PropertyChangedEventHandler PropertyChanged;

        #endregion

        public void BeginEdit()
        {
            if (isEdit) return;
            isEdit = true;
            this.backup = this.MemberwiseClone() as Person;
        }


        public void CancelEdit()
        {
            if (!this.isEdit)
                return;
            isEdit = false;
            this.Name = this.backup.Name;
            this.Salary = this.backup.Salary;
        }

        public void EndEdit()
        {
            if (this.isEdit == false)
                return;
            this.isEdit = false;
            this.backup = null;
        }



        private bool isEdit;
        private Person backup;
    }


    public class PerColl : ObservableCollection<Person> { }

    public class Kid : IEditableObject,INotifyPropertyChanged
    {
        public int Id
        {
            get { return _id; }
            set
            {
                this._id = value;
                if (PropertyChanged != null)
                    this.PropertyChanged(this, new PropertyChangedEventArgs("Id"));

            }

        }
        public string Name
        {
            get { return _name; }
            set
            {
                this._name = value;
                if (PropertyChanged != null)
                    this.PropertyChanged(this, new PropertyChangedEventArgs("Name"));

            }
        }
        public Person Parent { get; set; }

        public Kid()
        {
            this.Id = 12345;
            this.Name = "kidname";
        }

        public Kid(int id, string name = "kidname")
        {
            this.Id = 12345;
            this.Name = name;
        }

        #region IEditableObject Members

        public void BeginEdit()
        {
            if (isEdit) return;
            isEdit = true;
            this.backup = this.MemberwiseClone() as Kid;
        }

        public void CancelEdit()
        {
            if (!this.isEdit)
                return;
            isEdit = false;
            this.Id = backup.Id;
        }

        public void EndEdit()
        {
            if (this.isEdit == false)
                return;
            this.isEdit = false;
            this.backup = null;
        }

        #endregion
        private int _id;
        private string _name;
        private bool isEdit;
        private Kid backup;

        #region INotifyPropertyChanged Members

        public event PropertyChangedEventHandler PropertyChanged;

        #endregion
    }

}
}

Xaml 代码:

<Window x:Class="WpfApplication7.MainWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    Title="MainWindow" Height="350" Width="525">
<DockPanel LastChildFill="True" Margin="-1,5,1,-5">
    <Button Content="Add" Height="30" Click="Button_Click" DockPanel.Dock="Top"/>
    <DataGrid ItemsSource="{Binding Path=People}"
              AutoGenerateColumns="False"
              SelectionUnit="CellOrRowHeader"
              RowDetailsVisibilityMode="Visible">
        <DataGrid.Columns>
            <DataGridTextColumn Header="Salary" Binding="{Binding Path=Salary,StringFormat='{}{0:#,0}'}"/>
            <DataGridTextColumn Header="Name" Binding="{Binding Path=Name}"/>
            <DataGridCheckBoxColumn Header="IsLocked" Binding="{Binding Path=IsLocked,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged}"/>
        </DataGrid.Columns>

        <DataGrid.RowDetailsTemplate>
            <DataTemplate>
                <DataGrid
                    ItemsSource="{Binding Path=Kids}"
              AutoGenerateColumns="False"
                    IsReadOnly="{Binding Path=IsLocked}"
              SelectionUnit="CellOrRowHeader">
                    <DataGrid.Columns>
                        <DataGridTextColumn Header="Id" Binding="{Binding Path=Id,StringFormat='{}{0:#,0}'}"/>
                        <DataGridTextColumn Header="Name" Binding="{Binding Path=Name}"/>
                    </DataGrid.Columns>
                </DataGrid>
            </DataTemplate>

        </DataGrid.RowDetailsTemplate>
    </DataGrid>
</DockPanel>

4

2 回答 2

0

尝试在 Id 列上设置 TargetNullValue 并使 Kid.Id 为可为空的 int ,那么您的问题应该得到解决。

有关 TargetNUllValue 属性的信息,请参阅:http: //msdn.microsoft.com/en-us/library/system.windows.data.bindingbase.targetnullvalue.aspx

后面的代码:

public class Kid : IEditableObject, INotifyPropertyChanged
    {
        public int? Id
        {
            get { return _id; }
            set
            {
                this._id = value;
                if (PropertyChanged != null)
                    this.PropertyChanged(this, new PropertyChangedEventArgs("Id"));

            }

        }
        public string Name
        {
            get { return _name; }
            set
            {
                this._name = value;
                if (PropertyChanged != null)
                    this.PropertyChanged(this, new PropertyChangedEventArgs("Name"));

            }
        }
        public Person Parent { get; set; }

        public Kid()
        {
            this.Id = 12345;
            this.Name = "kidname";
        }

        public Kid(int? id, string name = "kidname")
        {
            this.Id = id;
            this.Name = name;
        }

        #region IEditableObject Members

        public void BeginEdit()
        {
            if (isEdit) return;
            isEdit = true;
            this.backup = this.MemberwiseClone() as Kid;
        }

        public void CancelEdit()
        {
            if (!this.isEdit)
                return;
            isEdit = false;
            this.Id = backup.Id;
        }

        public void EndEdit()
        {
            if (this.isEdit == false)
                return;
            this.isEdit = false;
            this.backup = null;
        }

        #endregion
        private int? _id;
        private string _name;
        private bool isEdit;
        private Kid backup;

        #region INotifyPropertyChanged Members

        public event PropertyChangedEventHandler PropertyChanged;

        #endregion
    }

WPF:(确保您拥有系统:添加了参考

<Window x:Class="WpfApplication3.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:system="clr-namespace:System;assembly=mscorlib"
        Title="MainWindow" Height="350" Width="525">
    <DockPanel LastChildFill="True" Margin="-1,5,1,-5">
        <Button Content="Add" Height="30" Click="Button_Click" DockPanel.Dock="Top"/>
        <DataGrid ItemsSource="{Binding Path=People}"
              AutoGenerateColumns="False"
              SelectionUnit="CellOrRowHeader"
              RowDetailsVisibilityMode="Visible">
            <DataGrid.Columns>
                <DataGridTextColumn Header="Salary" Binding="{Binding Path=Salary,StringFormat='{}{0:#,0}'}"/>
                <DataGridTextColumn Header="Name" Binding="{Binding Path=Name}"/>
                <DataGridCheckBoxColumn Header="IsLocked" Binding="{Binding Path=IsLocked,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged}"/>
            </DataGrid.Columns>

            <DataGrid.RowDetailsTemplate>
                <DataTemplate>
                    <DataGrid
                    ItemsSource="{Binding Path=Kids}"
              AutoGenerateColumns="False"
                    IsReadOnly="{Binding Path=IsLocked}"
              SelectionUnit="CellOrRowHeader">
                        <DataGrid.Columns>
                            <DataGridTextColumn Header="Id" Binding="{Binding Path=Id,StringFormat='{}{0:#,0}', TargetNullValue={x:Static system:String.Empty}}"/>
                            <DataGridTextColumn  Header="Name" Binding="{Binding Path=Name}"/>
                        </DataGrid.Columns>
                    </DataGrid>
                </DataTemplate>

            </DataGrid.RowDetailsTemplate>
        </DataGrid>
    </DockPanel>
</Window>
于 2013-08-03T15:08:34.647 回答
0

您可以将 Id 的类型从int更改为string

如果 Id 列是唯一的,则必须将其设为ReadOnly

或者你可以通过 IvalueConverter 处理它

按照这个

创建一个类并实现 IValueConverter

public class IntConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        return value;
    }

    public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        if (value == null || value.ToString()=="")
            return 0;
        return value;
    }
}

然后将 Coverter 作为资源添加到您的窗口

xmlns:local="clr-namespace:WpfApplication1"

<Window.Resources>
    <local:IntConverter x:Key="converter" />
</Window.Resources>

最后将转换器添加到您的Id

<DataGridTextColumn Header="Id" Binding="{Binding Path=Id,StringFormat='{}{0:#,0}',Converter={StaticResource converter}}" />
于 2013-07-30T08:16:51.873 回答