gpt4 book ai didi

c# - Windows UI 中的 ListView 缓存?

转载 作者:太空狗 更新时间:2023-10-29 23:50:26 27 4
gpt4 key购买 nike

我有以下问题:我有一个 Listview,其中每个项目都是一个自定义控件(“DisruptionIcon”),它显示一个图标,具体取决于为自定义控件设置的属性。图标可以是多种形状中的一种(此处示例中只有“无”和“方形”)。现在的问题是,如果我在 ListView 中向下滚动并在备份之后,图标是错误的。结果是:

在向下滚动之前:

Before Scrolling down

上下滚动后:

enter image description here

资源是:

主页.xaml:

<Page
x:Class="App1.MainPage"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:App1"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d"
Name="MainPg"
>

<Grid Background="{ThemeResource ApplicationPageBackgroundThemeBrush}">
<ListView
Width="200"
ItemsSource="{Binding LineList, ElementName=MainPg}"
>
<ListView.ItemTemplate>
<DataTemplate>
<Grid>
<local:DisruptionIcon
DisplayName="{Binding DisplayName}"
IconType="{Binding DisplayStyle}"
Color1="{Binding Color1}"
Color2="{Binding Color2}"
/>
<TextBlock Text="{Binding DisplayName}" Foreground="Red"/>
</Grid>
</DataTemplate>
</ListView.ItemTemplate>
</ListView>
</Grid>

MainPage.xaml.cs:

public sealed partial class MainPage : Page
{
public MainPage()
{
LineList = new ObservableCollection<ServerLineDefinition>();

for (byte i = 0; i <= 250; i++)
{
var ds = new ServerLineDefinition("ID " + i, "FFFFFF", "000000", 1, i);

LineList.Add(ds);
}

InitializeComponent();
}

public ObservableCollection<ServerLineDefinition> LineList { get; set; }
}

我的自定义控件是这样的:DisruptionIcon.xaml:

<UserControl
x:Class="App1.DisruptionIcon"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="using:App1"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
Name="mainUc" mc:Ignorable="d"
d:DesignHeight="34" d:DesignWidth="80"
Width="80" Height="34"
>
<Grid>
<local:DisplayIconTemplateSelector Content="{Binding IconType, ElementName=mainUc}">
<local:DisplayIconTemplateSelector.StyleNone>
<DataTemplate>
<Grid/>
</DataTemplate>
</local:DisplayIconTemplateSelector.StyleNone>
<local:DisplayIconTemplateSelector.StyleSquare>
<DataTemplate>
<Rectangle
Width="80"
Height="32"
Fill="{Binding Color2, ElementName=mainUc}"
VerticalAlignment="Top"
HorizontalAlignment="Left"
StrokeThickness="2"
Stroke="{Binding Color1, ElementName=mainUc}"
/>
</DataTemplate>
</local:DisplayIconTemplateSelector.StyleSquare>
</local:DisplayIconTemplateSelector>

<Viewbox
Stretch="Uniform"
HorizontalAlignment="Stretch"
VerticalAlignment="Stretch"
Margin="8,0"
>
<TextBlock
Text="{Binding DisplayName, ElementName=mainUc}"
Foreground="{Binding Color1, ElementName=mainUc}"
Height="auto"
VerticalAlignment="Top"
TextWrapping="NoWrap"
FontWeight="Bold"
TextAlignment="Center"
Margin="0,-2,0,0"
/>
</Viewbox>
</Grid>

这是 disruptionicon.xaml.csDisruptionIcon 的代码,其中基本上属性仅由 DependencyProperties 绑定(bind),颜色将转换为 SolidColorBrush 如果它们以字符串形式提供。由于应用程序设计,这是我需要的东西:

public sealed partial class DisruptionIcon
{
public DisruptionIcon()
{
Color1 = new SolidColorBrush(Colors.White);
Color2 = new SolidColorBrush(Colors.Black);
IconType = DisplayStyle.None;
DisplayName = "";
InitializeComponent();
}

#region Color1
/// <summary>
/// Used for the View of multiple Icons. Contains an IconId and the text to show for every Entry
/// </summary>
public object Color1
{
get { return (object)GetValue(DisruptionColor1Property); }
set { SetValue(DisruptionColor1Property, value); }
}

// Using a DependencyProperty as the backing store for ItemSource. This enables animation, styling, binding, etc...
public static readonly DependencyProperty DisruptionColor1Property =
DependencyProperty.Register("Color1", typeof(object), typeof(DisruptionIcon), new PropertyMetadata(null, DisruptionColor1PropertyCallback));

public static void DisruptionColor1PropertyCallback(DependencyObject dp, DependencyPropertyChangedEventArgs args)
{
var iconColor1 = ((DisruptionIcon)dp).Color1;

if (iconColor1 == null || iconColor1 is SolidColorBrush)
return;

//Convert Colors
if (iconColor1 is string)
iconColor1 = ConvertToSolidColorBrush((string)iconColor1);

((DisruptionIcon)dp).Color1 = iconColor1;
}
#endregion

#region Color2
/// <summary>
/// Used for the View of multiple Icons. Contains an IconId and the text to show for every Entry
/// </summary>
public object Color2
{
get { return (object)GetValue(DisruptionColor2Property); }
set { SetValue(DisruptionColor2Property, value); }
}

// Using a DependencyProperty as the backing store for ItemSource. This enables animation, styling, binding, etc...
public static readonly DependencyProperty DisruptionColor2Property =
DependencyProperty.Register("Color2", typeof(object), typeof(DisruptionIcon), new PropertyMetadata(null, DisruptionColor2PropertyCallback));

public static void DisruptionColor2PropertyCallback(DependencyObject dp, DependencyPropertyChangedEventArgs args)
{
var iconColor2 = ((DisruptionIcon)dp).Color2;

if (iconColor2 == null || iconColor2 is SolidColorBrush)
return;

//Convert Colors
if (iconColor2 is string)
iconColor2 = ConvertToSolidColorBrush((string)iconColor2);

((DisruptionIcon)dp).Color2 = iconColor2;
}
#endregion

#region IconType
/// <summary>
/// Used for the View of multiple Icons. Contains an IconId and the text to show for every Entry
/// </summary>
public object IconType
{
get { return (DisplayStyle)GetValue(DisruptionDisplayStyleProperty); }
set { SetValue(DisruptionDisplayStyleProperty, value); }
}

// Using a DependencyProperty as the backing store for ItemSource. This enables animation, styling, binding, etc...
public static readonly DependencyProperty DisruptionDisplayStyleProperty =
DependencyProperty.Register("IconType", typeof(object), typeof(DisruptionIcon), new PropertyMetadata(DisplayStyle.None, DisruptionDisplayStylePropertyCallback));

public static void DisruptionDisplayStylePropertyCallback(DependencyObject dp, DependencyPropertyChangedEventArgs args)
{
var iconDisplayStyle = ((DisruptionIcon)dp).IconType;

if (args.NewValue is Int32)
{
((DisruptionIcon)dp).IconType = (DisplayStyle)args.NewValue;
}


}

#endregion

#region DisplayName
/// <summary>
/// Used for the View of multiple Icons. Contains an IconId and the text to show for every Entry
/// </summary>
public string DisplayName
{
get { return (string)GetValue(DisruptionDisplayNameProperty); }
set { SetValue(DisruptionDisplayNameProperty, value); }
}

// Using a DependencyProperty as the backing store for ItemSource. This enables animation, styling, binding, etc...
public static readonly DependencyProperty DisruptionDisplayNameProperty =
DependencyProperty.Register("DisplayName", typeof(string), typeof(DisruptionIcon), new PropertyMetadata(null, DisruptionDisplayNamePropertyCallback));

public static void DisruptionDisplayNamePropertyCallback(DependencyObject dp, DependencyPropertyChangedEventArgs args)
{
var iconDisplayName = ((DisruptionIcon)dp).DisplayName;

if (iconDisplayName == null)
return;

((DisruptionIcon)dp).DisplayName = iconDisplayName;
}
#endregion


/// <summary>
/// Converts a ColorCode (i.e. FF8899) to SolidColorBrush
/// </summary>
/// <param name="colorCode">Six-Digit Hex-Code of the Color</param>
/// <returns></returns>
private static SolidColorBrush ConvertToSolidColorBrush(string colorCode)
{
if (colorCode != null && colorCode.Length == 6)
{
return new SolidColorBrush(Color.FromArgb(255,
Convert.ToByte(colorCode.Substring(0, 2), 16),
Convert.ToByte(colorCode.Substring(2, 2), 16),
Convert.ToByte(colorCode.Substring(4), 16)));
}

return new SolidColorBrush(Colors.Black);

}
}

我的模板选择器 DisplayIconTemplateSelector.cs:

public class DisplayIconTemplateSelector : ContentControl
{
protected override void OnContentChanged(object oldContent, object newContent)
{
base.OnContentChanged(oldContent, newContent);

ContentTemplate = SelectTemplate(newContent, this);
}

public DataTemplate StyleNone { get; set; }
public DataTemplate StyleSquare { get; set; }

public DataTemplate SelectTemplate(object item, DependencyObject container)
{
var quoteItem = (DisplayStyle)item;

switch (quoteItem)
{
default:
return StyleNone;
case DisplayStyle.Square:
return StyleSquare;
}
}
}

最后是我的 ServerLineDefinition 类:

public class ServerLineDefinition : INotifyPropertyChanged
{
public ServerLineDefinition() { }

public ServerLineDefinition(
string displayName,
string backgroundColor,
string foregroundColor,
int displayStyle,
int id)
{
DisplayName = displayName;
Color2 = backgroundColor;
Color1 = foregroundColor;
DisplayStyle = displayStyle;
Id = id;
}

public int Id { get; set; }
public string DisplayName { get; set; }
public int DisplayStyle { get; set; }


/// <summary>
/// RGB-Value for BackgroundColor
/// </summary>
public string Color2 { get; set; }

/// <summary>
/// RGB-Value for ForegroundColor
/// </summary>
public string Color1 { get; set; }


#region PropertyChanged

public event PropertyChangedEventHandler PropertyChanged; //To Update Content on the Form

/// <summary>
/// Helper for Triggering PropertyChanged
/// </summary>
/// <param name="triggerControl">The Name of the Property to update</param>
private void RaisePropertyChanged(string triggerControl)
{
if (PropertyChanged != null)
{
PropertyChanged(this, new PropertyChangedEventArgs(triggerControl));
}
}

#endregion
}

在我看来,ListView 的缓存存在问题。如果我将 ListView 替换为包裹在 ScrollViewer 中的 ItemsControl,则问题不存在,但会占用更多内存并需要更多时间载入。此外,如果您取消注释作为注释添加到 MainPage.xaml 中的 TextBlock,您会在每一行中看到正确的 ID,但会看到错误的图像,如截图。

编辑:如果我将任何控件放在 ScrollViewer 中,那么它会很慢,但可以工作。如果我将 DisruptionIcon.xaml 的整个代码直接放到 MainPage 中引用 DirutpionIcon 的位置,它也可以工作。

最佳答案

尝试为 ItemsControl 使用此样式 setter ,关键字“虚拟化”:

    <Setter Property="ItemsPanel">
<Setter.Value>
<ItemsPanelTemplate>
<ItemsStackPanel Orientation="Vertical"/>
</ItemsPanelTemplate>
</Setter.Value>
</Setter>
<Setter Property="Template">
<Setter.Value>
<ControlTemplate TargetType="ItemsControl">
<Border x:Name="LayoutRoot">
<ScrollViewer>
<ItemsPresenter/>
</ScrollViewer>
</Border>
</ControlTemplate>
</Setter.Value>
</Setter>

并在您的DisplayNameDependencyProperty 上,更新为:

public static void DisruptionDisplayNamePropertyCallback(DependencyObject dp, DependencyPropertyChangedEventArgs args)
{
var iconDisplayName = (string)e.NewValue;
((DisruptionIcon)dp).DisplayName = iconDisplayName;
}

关于c# - Windows UI 中的 ListView 缓存?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31817050/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com