Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

ScrollViewer or ItemsRepeater does not render items added to the top #28

Open
Hunv opened this issue Oct 11, 2024 · 5 comments
Open

ScrollViewer or ItemsRepeater does not render items added to the top #28

Hunv opened this issue Oct 11, 2024 · 5 comments
Labels
bug Something isn't working

Comments

@Hunv
Copy link

Hunv commented Oct 11, 2024

Describe the bug

I have this:

<ScrollViewer
  ScrollViewer.VerticalScrollBarVisibility= "Auto"
  Grid.Row= "0"
  >
  <cntrl:ItemsRepeater
      ItemsSource = "{Binding MyItemList}"
      Background= "{DynamicResource SiteBackgroundBrush}"
    >
    <ItemsRepeater.ItemTemplate >
      <DataTemplate DataType= "m:MyModel" >
        <Border
          BorderThickness= "2"
          CornerRadius= "5"
          BorderBrush= "{DynamicResource ItemBackgroundBrush}"
          Background= "{DynamicResource ItemBackgroundBrush}"
          Margin= "5,5,5,0"
          >
          <Grid Margin= "0,5,0,0" >
            ...            
          </Grid>
        </Border>
      </DataTemplate>
    </ItemsRepeater.ItemTemplate>
  </cntrl:ItemsRepeater>
</ScrollViewer>

The MyItemList is an ObservableCollection.
When I add a new Item to the Top of the list via MyItemList.Insert(0, ...) and run a this.RaisePropertyChanged(nameof(MyItemList));, the space of the new item in the ItemsRepeater is used but the item itself is not rendered. This does not happen from the very beginning but after some (not so long) time. I think shortly after when the first items runs out of the visible part of the Window.
If I scroll just a pixel in the ScrollViewer, everything is where and how it should be immediately.
This does not happen, if the ScrollViewer is not involved.

This is how it looks like:
image

To Reproduce

Create a new Avalonia Project from scratch with Desktop enabled. Leave everything else at default.
Make the following changes:

  1. Add the Avalonia.ItemsRepeater Nuget to the main project
  2. Replace code in MainViewModel.cs with this:
using System.Collections.ObjectModel;
using System;
using ReactiveUI;

namespace AvaloniaApplication1.ViewModels;

public class MainViewModel : ViewModelBase
{
    public ObservableCollection<MyModel> MyItemList { get; set; } = new ObservableCollection<MyModel>();
    private readonly System.Timers.Timer _TmrDisruptionUpdate = new(1000);

    public MainViewModel() 
    { 
        MyItemList.Add(new MyModel() { Text = "foobar"});
        MyItemList.Add(new MyModel() { Text = "Hello World" });
        MyItemList.Add(new MyModel() { Text = "John Doe" });
        MyItemList.Add(new MyModel() { Text = "xxxxx" });

        _TmrDisruptionUpdate.Elapsed += _TmrDisruptionUpdate_Elapsed;
        _TmrDisruptionUpdate.Start();
    }

    private void _TmrDisruptionUpdate_Elapsed(object? sender, System.Timers.ElapsedEventArgs e)
    {
        MyItemList.Insert(0, new MyModel() { Text = DateTime.Now.ToString() });        
        this.RaisePropertyChanged(nameof(MyItemList));
    }
}
  1. Create a new class called MyModel.cs with the following content:
namespace AvaloniaApplication1.ViewModels
{
    public class MyModel
    {
        public string Text { get; set; }
    }
}
  1. Edit the MainView.axaml as follows:
<UserControl xmlns="https://github.com/avaloniaui"
             xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
             xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
             xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
             xmlns:vm="clr-namespace:AvaloniaApplication1.ViewModels"
             xmlns:cntrl="using:Avalonia.Controls"
             mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450"
             x:Class="AvaloniaApplication1.Views.MainView"
             x:DataType="vm:MainViewModel">
  
  <ScrollViewer
    ScrollViewer.VerticalScrollBarVisibility="Auto"
    >
    <cntrl:ItemsRepeater
        ItemsSource="{Binding MyItemList}"
        Background="#222222"
        >
      <ItemsRepeater.ItemTemplate>
        <DataTemplate DataType="vm:MyModel">
          <Border
            BorderThickness="2"
            CornerRadius="5"
            BorderBrush="#888888"
            Background="#888888"
            Margin="5,5,5,0"
            >
            <Grid Margin="0,20">
              <TextBlock
                Text="{Binding Text}"
                />
            </Grid>
          </Border>
        </DataTemplate>
      </ItemsRepeater.ItemTemplate>
    </cntrl:ItemsRepeater>
  </ScrollViewer>
</UserControl>

I create a sample project with the things above:
ScrollViewerRenderIssue.zip

Expected behavior

Also later added items are rendered as the first items that were added.

Avalonia version

11.1.4

OS

Windows

Additional context

No response

@Hunv Hunv added the bug Something isn't working label Oct 11, 2024
@thevortexcloud
Copy link

FYI: ItemsRepeater is being deprecated soonish. It is however still somewhat supported right now.

https://github.com/AvaloniaUI/Avalonia.Controls.ItemsRepeater

@Hunv
Copy link
Author

Hunv commented Oct 12, 2024

FYI: ItemsRepeater is being deprecated soonish. It is however still somewhat supported right now.

https://github.com/AvaloniaUI/Avalonia.Controls.ItemsRepeater

I wasn't aware that the ItemsRepeater is deprecated.
OK, thank you for the hint. On the first view ItemsControl can be used as a drop-in replacement for the ItemsRepeater in this case. This also seems to fix the issue.
But it seems that the ItemsControl is much slower than the ItemsRepeater if the list is longer (i.e. 1000 items).

@thevortexcloud
Copy link

thevortexcloud commented Oct 13, 2024

I wasn't aware that the ItemsRepeater is deprecated.

It's not yet. It will be sometime after 12.0 releases though (no idea when that is). I will say you should still avoid using it for new developments though. But fixes and bug reports are technically still being accepted for it.

But it seems that the ItemsControl is much slower than the ItemsRepeater if the list is longer (i.e. 1000 items).

Do you have virtualisation enabled? The majority of ItemsRepeater panels are virtualised. ItemsControl however is not virtualised by default.

https://github.com/AvaloniaUI/Avalonia/blob/a04af7cebf3d49b81c228cbb6c18eb2e860ada58/src/Avalonia.Controls/ItemsControl.cs#L28-L32

If you change ItemsPanel on the ItemsControl to a VirtualizingStackPanel it will probably fix your issue.

@maxkatz6 maxkatz6 transferred this issue from AvaloniaUI/Avalonia Oct 13, 2024
@Hunv
Copy link
Author

Hunv commented Oct 14, 2024

Thank you very much for that hint. This replaces the ItemsRepeater completely - at least for me:

 <ScrollViewer
    ScrollViewer.VerticalScrollBarVisibility="Auto"
    >
    <ItemsControl
        ItemsSource="{Binding MyItemList}"
        >
      <ItemsControl.ItemTemplate>
        <DataTemplate DataType="vm:MyModel">
          <Border>
            ...
          </Border>
        </DataTemplate>
      </ItemsControl.ItemTemplate>
      <ItemsControl.ItemsPanel>
        <ItemsPanelTemplate>
          <VirtualizingStackPanel Orientation="Vertical" />
        </ItemsPanelTemplate>
      </ItemsControl.ItemsPanel>
    </ItemsControl>
  </ScrollViewer>

@thevortexcloud
Copy link

Glad I could help

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working
Projects
None yet
Development

No branches or pull requests

2 participants