1

我在 WPF 中的一个应用程序中遇到了一个非常奇怪的问题,其中很多屏幕与绑定和组合框配合得非常好。但其中之一给我带来了问题。

我创建了一个屏幕来为应用程序中定义的每个用户定义一个配置文件。所以它是一个 Listview,每一行都是一个标签(用户名)和一个带有配置文件列表的组合框。一切都是通过绑定来定义的。

这是 ListView 的 XAML(我删除了样式):

<ListView Name="lv_UserProfils" ItemsSource="{Binding ListeEntites}" AlternationCount="2"
ScrollViewer.VerticalScrollBarVisibility="Auto" ScrollViewer.HorizontalScrollBarVisibility="Auto">
    <ListView.View>
        <GridView>
            <GridViewColumn Header="Nom d'utilisateur" Width="250">
                <GridViewColumn.CellTemplate>
                    <DataTemplate>
                        <Border Height="25">
                            <TextBlock Text="{Binding UserLogin}" Width="Auto" />
                        </Border>
                    </DataTemplate>
                </GridViewColumn.CellTemplate>
            </GridViewColumn>
            <GridViewColumn Header="Profil" Width="Auto">
                <GridViewColumn.CellTemplate>
                    <DataTemplate>
                        <ComboBox 
                                        ItemsSource="{Binding DataContext.ListeProfils, ElementName=lv_UserProfils}" 
                                        DisplayMemberPath="LibProfil" SelectedValuePath="IdProfil" 
                                        SelectedValue="{Binding Profil.IdProfil}" 
                                        SelectedItem="{Binding Profil}" Width="200" />
                    </DataTemplate>
                </GridViewColumn.CellTemplate>
            </GridViewColumn>
        </GridView>
    </ListView.View>
</ListView>

DataContext是一个自定义 ViewModel 类的实例,它提供了一个ObservableCollection<UserBE>名为 ListeEntites

UserBE或多或少:

public sealed class UserBE
{
public bool IsAdmin { get; set; }
public bool IsUpdateGranted { get; set; }

private string _userLogin;
public string UserLogin
{
    get { return _userLogin; }
    set { _userLogin = value; OnPropertyChanged("UserLogin"); }
}

private ProfilBE _profil;
public ProfilBE Profil
{
    get { return _profil; }
    set
    {
        _profil = value;
        OnPropertyChanged("Profil");
    }
}
}

并且ProfilBE

public sealed class ProfilBE
{
    public long IdProfil { get; set; }

    private string _codProfil;
    public string CodProfil
    {
        get { return _codProfil; }
        set { _codProfil = value; OnPropertyChanged("CodProfil"); }
    }

    private string _libProfil;
    public string LibProfil
    {
        get { return _libProfil; }
        set { _libProfil = value; OnPropertyChanged("LibProfil"); }
    }
}

这是我的问题:
用户列表很长,所以有一个滚动条。我可以随心所欲地向下滚动,但只要我向上滚动(但前提是我向下滚动足够多),所有未显示的组合框一旦出现在屏幕上就会开始被清除。

有趣的事实 :

  • 当我滚动时,Profilsetter 会不断地调用与显示的行关联的对象。我不知道为什么(没有理由,该Profil属性已经定义)
  • 在某一时刻,如果我向上滚动,我会得到很多这样的异常,并且Profilsetter 开始接收null

.

System.Windows.Data Error: 23 : Cannot convert 'BanquePrivee.AssuranceVie.Net.BE.ProfilBE' from type 'ProfilBE' to type 'System.Int64' for 'fr-FR' culture with default conversions; consider using Converter property of Binding. NotSupportedException:'System.NotSupportedException: Int64Converter cannot convert from BanquePrivee.AssuranceVie.Net.BE.ProfilBE.
   at System.ComponentModel.TypeConverter.GetConvertFromException(Object value)
   at System.ComponentModel.TypeConverter.ConvertFrom(ITypeDescriptorContext context, CultureInfo culture, Object value)
   at System.ComponentModel.BaseNumberConverter.ConvertFrom(ITypeDescriptorContext context, CultureInfo culture, Object value)
   at MS.Internal.Data.DefaultValueConverter.ConvertHelper(Object o, Type destinationType, DependencyObject targetElement, CultureInfo culture, Boolean isForward)'

System.Windows.Data Error: 7 : ConvertBack cannot convert value 'BanquePrivee.AssuranceVie.Net.BE.ProfilBE' (type 'ProfilBE'). BindingExpression:Path=Profil.IdProfil; DataItem='UserBE' (HashCode=59629589); target element is 'ComboBox' (Name=''); target property is 'SelectedValue' (type 'Object') NotSupportedException:'System.NotSupportedException: Int64Converter cannot convert from BanquePrivee.AssuranceVie.Net.BE.ProfilBE.
   at MS.Internal.Data.DefaultValueConverter.ConvertHelper(Object o, Type destinationType, DependencyObject targetElement, CultureInfo culture, Boolean isForward)
   at MS.Internal.Data.ObjectTargetConverter.ConvertBack(Object o, Type type, Object parameter, CultureInfo culture)
   at System.Windows.Data.BindingExpression.ConvertBackHelper(IValueConverter converter, Object value, Type sourceType, Object parameter, CultureInfo culture)'

这很明显SelectedValue="{Binding Profil.IdProfil}"是问题所在,但我不明白为什么。
我不明白为什么在某些时候它会尝试IdProfilProfilBE. 我不应该在那里使用转换器。
我做了很多测试,数据似乎很好(不应该有空值等)

有人可以指出我做错了什么吗?

4

2 回答 2

1

我认为这与设置SelectedValueSelectedItem. 这两个属性都做同样的事情:它们设置选定的项目。但是一个Value根据是什么SelectedValuePath来设置它,另一个只是将它设置为ItemsSource.

我猜 WPF 在某处感到困惑,并试图将SelectedValue(它是 an int)设置SelectedItem为 type ProfilBE,并且抛出异常,因为 aProfilBE不能强制转换为 int。

但无论如何,要修复它,请尝试删除SelectedItemComboBox 中的绑定

<ComboBox ItemsSource="{Binding DataContext.ListeProfils, ElementName=lv_UserProfils}" 
    DisplayMemberPath="LibProfil" SelectedValuePath="IdProfil" 
    SelectedValue="{Binding Profil.IdProfil}" 
    Width="200" />
于 2012-09-27T18:56:43.777 回答
0

WPF 正在虚拟化当前未显示的对象。但是当我尝试向上滚动时,它似乎试图做一些对已虚拟化并即将再次出现在屏幕上的项目不起作用的事情。

我使用的解决方案是使用VirtualizingStackPanel.IsVirtualizing="False". 它对性能的影响很小,但现在它可以工作了。

<ListView Name="lv_UserProfils" ItemsSource="{Binding ListeEntites}" AlternationCount="2" VirtualizingStackPanel.IsVirtualizing="False"
        ScrollViewer.VerticalScrollBarVisibility="Auto" ScrollViewer.HorizontalScrollBarVisibility="Auto">
    <ListView.View>
        [...]
    </ListView.View>
</ListView>
于 2012-09-28T14:44:54.727 回答