在我的视图模型中覆盖OnActivate()期间,我需要调用GetView()来聚焦一个元素。 当我在之前激活视图后执行此操作时,它很好。 但是当我称之为第一次激活时,它失败了。
我能够通过在ConductorBaseWithActiveItem.ChangeActiveItem中交换几行来使其工作。 原文如下:
protected virtual void ChangeActiveItem(T newItem, bool closePrevious) {
ScreenExtensions.TryDeactivate(activeItem, closePrevious);
newItem = EnsureItem(newItem);
if(IsActive)
ScreenExtensions.TryActivate(newItem);
activeItem = newItem;
NotifyOfPropertyChange("ActiveItem");
OnActivationProcessed(activeItem, true);
}
以及我的更改:
protected virtual void ChangeActiveItem(T newItem, bool closePrevious) {
ScreenExtensions.TryDeactivate(activeItem, closePrevious);
newItem = EnsureItem(newItem);
activeItem = newItem;
NotifyOfPropertyChange("ActiveItem");
if (IsActive)
ScreenExtensions.TryActivate(newItem);
OnActivationProcessed(activeItem, true);
}
这似乎有效。 通知"活动项"已更改会触发代码以加载和缓存视图。 然后 ScreenExtensions.TryActivate 调用我的 OnActivate 覆盖。
问:我没有注意到这样做有任何问题,但我很好奇是否有人比我更了解这种变化会产生什么影响?
谢谢!
尝试的一件事是覆盖Caliburn的OnViewAttached
方法并尝试将其集中在那里。话虽如此,在 MVVM 中,焦点更像是视图关注点,因此如果可能,应将该逻辑从 ViewModel 移动到视图。
的一种方法是创建附加行为(您将需要对Microsoft.Expression.Interactions
程序集的引用):
public class FocusWhenVisibleBehavior : Behavior<FrameworkElement>
{
protected override void OnAttached()
{
this.AssociatedObject.Loaded += this.Loaded;
this.AssociatedObject.IsVisibleChanged += this.VisibleChanged;
}
protected override void OnDetaching()
{
this.AssociatedObject.Loaded -= this.Loaded;
this.AssociatedObject.IsVisibleChanged -= this.VisibleChanged;
}
private void Loaded(object sender, RoutedEventArgs e)
{
this.TryFocus();
}
private void VisibleChanged(object sender, DependencyPropertyChangedEventArgs e)
{
this.TryFocus();
}
private void TryFocus()
{
if (this.AssociatedObject.IsLoaded && this.AssociatedObject.IsVisible)
{
// Focus the control
this.AssociatedObject.Focus();
}
}
}
并将该行为附加到您想要关注的任何控件:
<Button>
<i:Interaction.Behaviors>
<b:FocusWhenVisibleBehavior/>
</i:Interaction.Behaviors>
</Button>