Install-Package Microsoft.Extensions.DependencyInjection;
Install-Package CommunityToolkit.mvvm;
//App.cs
using Microsoft.Extensions.DependencyInjection;
using System.Windows;
namespace WpfApp11
{
/// <summary>
/// Interaction logic for App.xaml
/// </summary>
public partial class App : Application
{
ServiceProvider serviceProvider;
protected override void OnStartup(StartupEventArgs e)
{
base.OnStartup(e);
ServiceCollection services = new ServiceCollection();
ConfigurationServices(services);
serviceProvider=services.BuildServiceProvider();
var mainWin = serviceProvider.GetRequiredService<MainWindow>();
if(mainWin != null)
{
mainWin.Show();
}
}
private void ConfigurationServices(ServiceCollection services)
{
services.AddSingleton<IIDService,IDService>();
services.AddSingleton<IISBNService,ISBNService>();
services.AddSingleton<INameService, NameService>();
services.AddSingleton<MainVM>();
services.AddSingleton<MainWindow>();
}
protected override void OnExit(ExitEventArgs e)
{
base.OnExit(e);
serviceProvider?.Dispose();
}
}
}
//Define IService and services
using CommunityToolkit.Mvvm.ComponentModel;
using System.Collections.ObjectModel;
using System.IO;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
namespace WpfApp11
{
/// <summary>
/// Interaction logic for MainWindow.xaml
/// </summary>
public partial class MainWindow : Window
{
public MainWindow()
{
InitializeComponent();
}
public MainWindow(MainVM vm)
{
InitializeComponent();
DataContext = vm;
this.SizeChanged+=MainWindow_SizeChanged;
}
private void MainWindow_SizeChanged(object sender, SizeChangedEventArgs e)
{
if (DataContext is MainVM vm)
{
var fe = this.Content as FrameworkElement;
if (fe==null)
{
return;
}
vm.GridWidth=fe.ActualWidth;
vm.GridHeight=fe.ActualHeight/2;
}
}
}
public partial class MainVM : ObservableObject
{
IIDService idService;
IISBNService iSBNService;
INameService nameService;
public MainVM(IIDService idServiceValue, IISBNService iSBNServiceValue,INameService nameServiceValue)
{
idService = idServiceValue;
iSBNService = iSBNServiceValue;
nameService= nameServiceValue;
InitData();
}
private async Task InitData()
{
var imgDir = @"../../../Images";
if(!Directory.Exists(imgDir))
{
return;
}
var imgs=Directory.GetFiles(imgDir);
if(imgs==null ||!imgs.Any())
{
return;
}
int imgsCount = imgs.Count();
BooksCollection=new ObservableCollection<Book>();
List<Book> booksList = new List<Book>();
await Task.Run(async () =>
{
for (int i = 0; i<1000000; i++)
{
booksList.Add(new Book()
{
ID=idService.GetID(),
Name=nameService.GetName(),
ISBN=iSBNService.GetISBN(),
ImgSource=GetImgSourceViaUrl(imgs[i%imgsCount])
});
if (i%100==0)
{
await PopulateBooksCollection(booksList);
}
}
if (booksList.Count>0)
{
await PopulateBooksCollection(booksList);
}
});
}
private async Task PopulateBooksCollection(List<Book> booksList)
{
var tempList = booksList.ToList();
booksList.Clear();
await Application.Current.Dispatcher.InvokeAsync(() =>
{
foreach (var bk in tempList)
{
BooksCollection.Add(bk);
}
StatusMsg=$"BooksCollection count:{BooksCollection.Count}";
});
await Task.Delay(1);
}
private ImageSource GetImgSourceViaUrl(string imgUrl)
{
BitmapImage bmi = new BitmapImage();
if(!File.Exists(imgUrl))
{
return bmi;
}
bmi.BeginInit();
bmi.UriSource=new Uri(imgUrl, UriKind.RelativeOrAbsolute);
bmi.EndInit();
if(bmi.CanFreeze)
{
bmi.Freeze();
}
return bmi;
}
[ObservableProperty]
private ObservableCollection<Book> booksCollection;
[ObservableProperty]
private double gridWidth;
[ObservableProperty]
private double gridHeight;
[ObservableProperty]
private string statusMsg;
}
public interface IIDService
{
int GetID();
}
public class IDService : IIDService
{
int id = 0;
public int GetID()
{
return ++id;
}
}
public interface IISBNService
{
string GetISBN();
}
public class ISBNService : IISBNService
{
int idx = 0;
public string GetISBN()
{
return $"ISBN_{++idx}_{Guid.NewGuid():N}";
}
}
public interface INameService
{
string GetName();
}
public class NameService : INameService
{
int idx = 0;
public string GetName()
{
return $"Name_{++idx}";
}
}
public class Book
{
public int ID { get; set; }
public string Name { get; set; }
public string ISBN { get; set; }
public ImageSource ImgSource { get; set; }
}
}
//xaml
<Window x:Class="WpfApp11.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
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:local="clr-namespace:WpfApp11"
WindowState="Maximized"
mc:Ignorable="d"
Title="{Binding StatusMsg,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged}"
Height="450" Width="800">
<Grid>
<ListBox ItemsSource="{Binding BooksCollection,Mode=TwoWay,UpdateSourceTrigger=PropertyChanged}"
VirtualizingPanel.IsVirtualizing="True"
VirtualizingPanel.VirtualizationMode="Recycling">
<ListBox.ItemTemplate>
<DataTemplate>
<Grid Width="{Binding DataContext.GridWidth,RelativeSource={RelativeSource AncestorType=Window}}"
Height="{Binding DataContext.GridHeight,RelativeSource={RelativeSource AncestorType=Window}}">
<Grid.Resources>
<Style TargetType="TextBlock">
<Setter Property="FontSize" Value="80"/>
<Setter Property="HorizontalAlignment" Value="Center"/>
<Setter Property="VerticalAlignment" Value="Center"/>
<Style.Triggers>
<Trigger Property="IsMouseOver" Value="True">
<Setter Property="FontSize" Value="130"/>
<Setter Property="Foreground" Value="Red"/>
</Trigger>
</Style.Triggers>
</Style>
</Grid.Resources>
<Grid.Background>
<ImageBrush ImageSource="{Binding ImgSource}"/>
</Grid.Background>
<Grid.RowDefinitions>
<RowDefinition/>
<RowDefinition/>
</Grid.RowDefinitions>
<Grid.ColumnDefinitions>
<ColumnDefinition/>
<ColumnDefinition/>
</Grid.ColumnDefinitions>
<TextBlock Text="{Binding ID}" Grid.Row="0" Grid.Column="0"/>
<TextBlock Text="{Binding Name}" Grid.Row="0" Grid.Column="1"/>
<TextBlock Text="{Binding ISBN}" Grid.Row="1" Grid.Column="0" Grid.ColumnSpan="2"/>
</Grid>
</DataTemplate>
</ListBox.ItemTemplate>
</ListBox>
</Grid>
</Window>
![image]()
![image]()
![image]()