WPF/SL: lazy loading TreeView
Posted on January 25, 2012 by Matthieu MEZIL
01/26/2012: Code update
Imagine the following scenario: you have a WCF service with two methods:
List<Customer> GetCustomers();
List<Order> GetOrders(int CustomerId);
You want a treeview with lazy loading in a WPF Window.
There is many way to do it.
I identify three main in my searches:
- you can use event on your treeview implemented in code-behind
- you can makes your TreeView control inheriting the framework one’s
- you can use all the logic on ViewModels and use binding
The last point is realized by adding a CustomerViewModel, having a collection of CustomerViewModel in the VM that encapsulated a Customer and adding IsExpanded property and add the logic of loading orders.
It’s a way often saw in the web and that seems a good way with MVVM for many developers but I think, IMHO, it is NOT a good way.
Indeed, what happens if under Orders, I want OrderDetails? You will add a new OrderViewModel class that encapsulates an Order and the CustomerViewModel class will have a collection of OrderViewModel?
I don’t want to make again my Model in my ViewModel.
I could use the ICustomTypeDescriptor (ICustomTypeProvider in SL) as I did here but I think that if this solution is interesting to add business logic on entity, it is not to add control logic.
I think that the lazy loading control logic should be encapsulated in a behavior and the ViewModel should just have the lazy loading WCF calls logic.
So, I use an ILazyLoader interface:
public interface ILazyLoader
{
string GetChildPropertyName(object obj);
bool IsLoaded(object obj);
void Load(object obj);
}
and an implementation of it using delegate:
public class LazyLoader : ILazyLoader
{
private Func<object, string> _getChildPropertyName;
private Func<object, bool> _isLoaded;
private Action<object> _load;
public LazyLoader(Func<object, string> getChildPropertyName, Func<object, bool> isLoaded, Action<object> load)
{
_getChildPropertyName = getChildPropertyName;
_isLoaded = isLoaded;
_load = load;
}
public string GetChildPropertyName(object obj)
{
return _getChildPropertyName(obj);
}
public bool IsLoaded(object obj)
{
return _isLoaded(obj);
}
public void Load(object obj)
{
_load(obj);
}
}
Then, in my ViewModel, I use the following code:
public class CustomerViewModel
{
private ObservableCollection<Customer> _customers;
public ObservableCollection<Customer> Customers
{
get
{
if (_customers == null)
{
_customers = new ObservableCollection<Customer>();
var customersService = new CustomerServiceClient();
EventHandler<GetCustomersCompletedEventArgs> serviceGetCustomersCompleted = null;
serviceGetCustomersCompleted = (sender, e) =>
{
customersService.GetCustomersCompleted -= serviceGetCustomersCompleted;
foreach (var ht in e.Result)
_customers.Add(ht);
};
customersService.GetCustomersCompleted += serviceGetCustomersCompleted;
customersService.GetCustomersAsync();
}
return _customers;
}
}
private ILazyLoader _lazyLoader;
public ILazyLoader LazyLoader
{
get { return _lazyLoader ?? (_lazyLoader = new LazyLoader(obj =>
{
if (obj is HardwareType)
return PropertyName.GetPropertyName((Expression<Func<HardwareType, object>>)(ht => ht.Hardwares));
return null;
}, obj => _loadedHardwareTypes.Contains((HardwareType)obj), obj => LoadHardwares((HardwareType)obj))); }
}
private List<Customer> _loadedCustomers = new List<Customer>();
private void LoadOrders(Customer c)
{
var customerService = new CustomerServiceClient();
c.Orders.Clear();
EventHandler<GetOrdersCompletedEventArgs> serviceGetOrdersCompleted = null;
serviceGetOrdersCompleted = (sender, e) =>
{
customerService.GetOrdersCompleted -= serviceGetOrdersCompleted;
foreach (var o in e.Result)
c.Orders.Add(o);
_loadedCustomers.Add(c);
};
customerService.GetOrdersCompleted += serviceGetCustomersCompleted;
customerService.GetOrdersAsync(c.Id);
}
}
Now, this is the code of my behavior:
public static class LazyLoadTreeViewItemBehavior
{
public static ILazyLoader GetLazyLoader(DependencyObject obj)
{
return (ILazyLoader)obj.GetValue(LazyLoaderProperty);
}
public static void SetLazyLoader(DependencyObject obj, ILazyLoader value)
{
obj.SetValue(LazyLoaderProperty, value);
}
public static readonly DependencyProperty LazyLoaderProperty =
DependencyProperty.RegisterAttached("LazyLoader", typeof(ILazyLoader), typeof(LazyLoadTreeViewItemBehavior), new PropertyMetadata(ApplyingLazyLoadingLogic));
private static void ApplyingLazyLoadingLogic(DependencyObject o, DependencyPropertyChangedEventArgs e)
{
var tvi = o as TreeViewItem;
if (tvi == null)
throw new InvalidOperationException();
ILazyLoader lazyLoader= GetLazyLoader(o);
PropertyInfo childrenProp;
if (lazyLoader == null)
return;
object itemValue = tvi.DataContext;
string childrenPropName = lazyLoader.GetChildPropertyName(itemValue);
if (childrenPropName == null || (childrenProp = itemValue.GetType().GetProperty(childrenPropName)) == null)
return;
IEnumerable children = (IEnumerable)childrenProp.GetValue(itemValue, null);
RoutedEventHandler tviExpanded = null;
RoutedEventHandler tviUnloaded = null;
tviExpanded = (sender, e2) =>
{
tvi.Expanded -= tviExpanded;
tvi.Unloaded -= tviUnloaded;
if (!lazyLoader.IsLoaded(itemValue))
{
lazyLoader.Load(itemValue);
tvi.Items.Clear();
tvi.ItemsSource = children;
}
};
tviUnloaded = (sender, e2) =>
{
tvi.Expanded -= tviExpanded;
tvi.Unloaded -= tviUnloaded;
};
if (!children.GetEnumerator().MoveNext())
{
tvi.ItemsSource = null;
tvi.Items.Add(new TreeViewItem());
}
tvi.Expanded += tviExpanded;
tvi.Unloaded += tviUnloaded;
}
}
The thing very interesting with it is the fact that my behavior is not dependent of my model or my ViewModel and can be used with other lazy loading TreeViews.
To do it, I just have to apply our behavior into our TreeView, what can be done in xaml:
<TreeView ItemsSource="{Binding Customers}">
<TreeView.ItemContainerStyle>
<Style TargetType="{x:Type TreeViewItem}">
<Setter Property="local:LazyLoadTreeViewItemBehavior.LazyLoader"
Value="{Binding DataContext.LazyLoader, RelativeSource={RelativeSource AncestorType=local:CustomersWindow}}" />
</Style>
</TreeView.ItemContainerStyle>
<TreeView.ItemTemplate>
<HierarchicalDataTemplate>
<HierarchicalDataTemplate.ItemTemplate>
<DataTemplate>
…
</DataTemplate>
</HierarchicalDataTemplate.ItemTemplate>
…
</HierarchicalDataTemplate>
</TreeView.ItemTemplate>
</TreeView>
I really like this way. What do you think about it?
Of course, I write my sample with WPF but it’s still true with SL.
Hope this helps…
This entry was posted in 13461, 7671, 8708. Bookmark the permalink.
WPF/SL: lazy loading TreeView的更多相关文章
- Angular2+typescript+webpack2(支持aot, tree shaking, lazy loading)
概述 Angular2官方推荐的应该是使用systemjs加载, 但是当我使用到它的tree shaking的时候,发现如果使用systemjs+rollup,只能打包成一个文件,然后lazy loa ...
- Lazyr.js – 延迟加载图片(Lazy Loading)
Lazyr.js 是一个小的.快速的.现代的.相互间无依赖的图片延迟加载库.通过延迟加载图片,让图片出现在(或接近))视窗才加载来提高页面打开速度.这个库通过保持最少选项并最大化速度. 在线演示 ...
- Can you explain Lazy Loading?
Introduction Lazy loading is a concept where we delay the loading of the object until the point wher ...
- [AngularJS] Lazy Loading modules with ui-router and ocLazyLoad
We've looked at lazy loading with ocLazyLoad previously, but what if we are using ui-router and want ...
- [AngularJS] Lazy loading Angular modules with ocLazyLoad
With the ocLazyLoad you can load AngularJS modules on demand. This is very handy for runtime loading ...
- iOS swift lazy loading
Why bother lazy loading and purging pages, you ask? Well, in this example, it won't matter too much ...
- Entity Framework加载相关实体——延迟加载Lazy Loading、贪婪加载Eager Loading、显示加载Explicit Loading
Entity Framework提供了三种加载相关实体的方法:Lazy Loading,Eager Loading和Explicit Loading.首先我们先来看一下MSDN对三种加载实体方法的定义 ...
- Lazy Loading | Explicit Loading | Eager Loading in EntityFramework and EntityFramework.Core
EntityFramework Eagerly Loading Eager loading is the process whereby a query for one type of entity ...
- EFCore Lazy Loading + Inheritance = 干净的数据表 (二) 【献给处女座的DB First程序猿】
前言 本篇是上一篇EFCore Lazy Loading + Inheritance = 干净的数据表 (一) [献给处女座的DB First程序猿] 前菜 的续篇.这一篇才是真的为处女座的DB Fi ...
随机推荐
- [转]在 javascript 按键事件中,按键值的对照表
转自:http://www.phpweblog.net/kiyone/archive/2007/04/19/1138.html 话说谁知道怎么能精简下word生成的html代码....好大啊... 字 ...
- 【Android开发】Android应用程序目录结构
原文:http://android.eoe.cn/topic/summary Android开发之旅:组件生命周期吴秦 Android开发之旅:HelloWorld项目的目录结构 * HelloWor ...
- MySQL 自定义函数CREATE FUNCTION实例
分享一个MySQL 自定义函数CREATE FUNCTION的实例.mysql> delimiter $$mysql> CREATE FUNCTION myFunction-> (i ...
- mybatis 对象关系映射例子
入门 http://legend2011.blog.51cto.com/3018495/908956 增删改 http://legend2011.blog.51cto.com/3018495/9130 ...
- Java Web(十一) 分页功能的实现
虽然现在有很多好用的框架,对分页进行支持,很简单的就把分页的效果做出来,但是如果自己手写是一个怎样的流程的?今天就来说说它,手动实现分页效果. --WH 一.分页的思路 首先我们得知道写分页代码时的思 ...
- [svc]java初步
J2EE.J2SE 和 J2ME 三者的区别有哪些? J2SE, 主要是桌面软件开发,包括swing 等.是后面2个的基础,但是我们不用它做桌面开发.只学习语法. J2EE,网站开发,servlet. ...
- ubuntu java 环境配置
下载oracle java sudo add-apt-repository ppa:webupd8team/java -y sudo apt-get update sudo apt-get insta ...
- angular学习笔记(三十一)-$location(1)
本篇介绍angular中的$location服务的基本用法,下一篇介绍它的复杂的用法. $location服务的主要作用是用于获取当前url以及改变当前的url,并且存入历史记录. 一. 获取url的 ...
- Spark SQL利器:cacheTable/uncacheTable【转】
转自:http://www.cnblogs.com/yurunmiao/p/4936583.html Spark相对于Hadoop MapReduce有一个很显著的特性就是“迭代计算”(作为一个Map ...
- Python 执行linux 命令
# !/usr/bin/env python # -*- coding: utf-8 -* import os def main(): file_list = os.popen("ls&qu ...