我ListBox
认为,绑定到一个动态增长的集合。我希望滚动位置跟随最后添加的项目(附加到列表底部)。如何使用 Caliburn.Micro 实现这一目标?
3 回答
另一种方法是使用事件聚合器将消息发布到视图。
就像是:
Aggregator.Publish(ItemAddedMessage<SomeItemType>(itemThatWasJustAdded));
在视图中:
public class SomeView : IHandle<ItemAddedMessage<SomeItemType>>
{
public void Handle(ItemAddedMessage<SomeItemType> message)
{
// Implement view specific behaviour here
}
}
这取决于您的要求,但至少视图负责显示问题,您仍然可以测试 VM
您也可以只在视图中实现代码 - 因为它似乎是一个视图问题(例如使用列表框提供的事件)
一种行为也会很有用,但可能与您的类型的耦合度较低 - 例如一种通用行为SeekAddedItemBehaviour
,它挂钩列表框事件以查找最后一项。不确定列表框是否公开了所需的事件,但值得一看
编辑:
好的,这可能会完全停止 - 你应该能够将此行为附加到列表框,它应该负责其余的:
public class ListBoxSeekLastItemBehaviour : System.Windows.Interactivity.Behavior<ListBox>
{
private static readonly DependencyProperty ItemsSourceWatcherProperty = DependencyProperty.Register("ItemsSourceWatcher", typeof(object), typeof(ListBoxSeekLastItemBehaviour), new PropertyMetadata(null, OnItemsSourceWatcherPropertyChanged));
private ListBox _listBox = null;
private static void OnItemsSourceWatcherPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
ListBoxSeekLastItemBehaviour source = d as ListBoxSeekLastItemBehaviour;
if (source != null)
source.OnItemsSourceWatcherPropertyChanged();
}
private void OnItemsSourceWatcherPropertyChanged()
{
// The itemssource has changed, check if it raises collection changed notifications
if (_listBox.ItemsSource is INotifyCollectionChanged)
{
// if it does, hook the CollectionChanged event so we can respond to items being added
(_listBox.ItemsSource as INotifyCollectionChanged).CollectionChanged += new NotifyCollectionChangedEventHandler(ListBoxSeekLastItemBehaviour_CollectionChanged);
}
}
void ListBoxSeekLastItemBehaviour_CollectionChanged(object sender, NotifyCollectionChangedEventArgs e)
{
if (e.Action == NotifyCollectionChangedAction.Add && e.NewItems.Count > 0)
{
// If an item was added seek it
ScrollIntoView(e.NewItems[0]);
}
}
protected override void OnAttached()
{
base.OnAttached();
// We've been attached - get the associated listbox
var box = this.AssociatedObject as ListBox;
if (box != null)
{
// Hold a ref
_listBox = box;
// Set a binding to watch for property changes
System.Windows.Data.Binding binding = new System.Windows.Data.Binding("ItemsSource") { Source = _listBox; }
// EDIT: Potential bugfix - you probably want to check the itemssource here just
// in case the behaviour is applied after the original ItemsSource binding has been evaluated - otherwise you might miss the change
OnItemsSourceWatcherPropertyChanged();
}
}
private void ScrollIntoView(object target)
{
// Set selected item and try and scroll it into view
_listBox.SelectedItem = target;
_listBox.ScrollIntoView(target);
}
}
您可能想稍微整理一下,并确保在更改CollectionChanged
时删除事件处理程序。ItemsSource
您也可能想调用它SeekLastAddedItemBehaviour
或SeekLastAddedItemBehavior
- 我倾向于保留美国拼写,因为它与 Microsoft 的拼写匹配。我认为SeekLastItem
听起来它会滚动到列表中的最后一项而不是最后添加的项目
您可以使用 GetView() 引用视图模型中的视图。这也将视图和视图模型耦合在一起。
var myView = GetView() as MyView;
myView.MyListBox.DoStuff
另一种选择是创建行为。 这是一个如何使用行为来扩展TreeView
视图模型的示例。这同样适用于ListBox
.
实际上,有一种更简单的方法可以实现这一点,而无需上述任何方法。
只需使用以下内容扩展您的列表框:
namespace Extensions.Examples {
public class ScrollingListBox : ListBox
{
protected override void OnItemsChanged(System.Collections.Specialized.NotifyCollectionChangedEventArgs e)
{
if (e.NewItems != null)
{
int newItemCount = e.NewItems.Count;
if (newItemCount > 0)
this.ScrollIntoView(e.NewItems[newItemCount - 1]);
base.OnItemsChanged(e);
}
}
}
}
然后在 Xaml 中,声明扩展类的位置,如下所示:
xmlns:Extensions="clr-namespace:Extensions.Examples"
当您创建列表框时,而不是使用
<Listbox></Listbox>
只需使用您的扩展类
<Extensions:ScrollingListBox></Extensions:ScrollingListBox>