问题:
在更改ContentControl的内容后,我无法进入时间点,应用了DataTemplateSelector中的DataTemplate,并且所有可视布局都已完成,即新内容已加载。
设定:
一个名为'KeyboardHost'的自定义控件,它扩展了 ContentControl 。
ContentControl.Content 通过值转换器绑定(使用多重绑定)到2个通知属性,该转换器将2个绑定属性组合到“KeyboardCriteria”类型的对象中。
'KeyboardCriteria'是一个公共类,但我也尝试将其作为FrameworkElement,Control和UserControl,以便我可以尝试挂钩Initialized,Loaded等事件。
ContentControl.ContentTemplateSelector 是一个自定义选择器类(如下所示),它返回一个 DataTemplate基于ContentControl.Content('KeyboardCriteria')。
ContentControl.ContentTemplateSelector的DataTemplates是选择器上的属性,在我的MainView的资源部分进行初始化和分配。
尝试:
我已附加/覆盖以下ContentControl事件:
Initialized
Loaded
OnContentChanged
OnContentTemplateChanged
我已附加/覆盖以下'KeyboardCriteria'(定义为FrameworkElement)事件:
Initialized
Loaded
OnApplyTemplate
OnTemplateChanged
TemplateDP callback
观察:
启动时:
KeyboardHost: OnTemplateChanged
KeyboardHost: ContentChanged
KeyboardCriteria: Initialized
KeyboardCriteria: Loaded
更改其中一个绑定条件属性(从而创建新的KeyboardCriteria对象)时:
KeyboardHost: ContentChanged
KeyboardCriteria: Initialized
N.B。 ContentControl.Content对象('KeyboardCriteria')缺少Loaded事件。
后续步骤:
我想我会废弃完全使用DataTemplateSelector并将选择逻辑构建到ContentControl中的想法,因为这已经是CustomControl了。我希望通过手动创建内容(并填充它),我可以避免使用我目前在选择逻辑中使用的DataTemplates,因为我怀疑这是问题的一部分。
...
代码示例:
MainViewModel:
公开“键盘”属性,该属性最初具有非空值。
的MainView:
<controls:KeyboardHost Grid.Row="0"
ContentTemplateSelector="{StaticResource KeyboardDataTemplateSelector}">
<ContentControl.Content>
<MultiBinding Converter="{StaticResource KeyboardCriteriaValueConverter}" Mode="OneWay">
<Binding Source="{x:Static properties:Settings.Default}" Path="Language" />
<Binding Path="Keyboard" />
</MultiBinding>
</ContentControl.Content>
</controls:KeyboardHost>
KeyboardCriteriaValueConverter:
仅转换方法,ConvertBack抛出NotImplementedException。
已简化逻辑以删除额外的逻辑以检查值类型,计数等。
public object Convert(object[] values, Type targetType, object parameter, CultureInfo culture)
{
return new KeyboardCriteria
{
Language = values[0],
Keyboard = values[1]
};
}
KeyboardDataTemplateSelector:
public class KeyboardDataTemplateSelector : DataTemplateSelector
{
//TEMPLATE PROPERTIES HERE - THESE ARE SET IN THE RESOURCE DEFINITION
public override DataTemplate SelectTemplate(object item, DependencyObject container)
{
var criteria = item as KeyboardCriteria;
//LOGIC TO RETURN THE APPROPRIATE KEYBOARD DATA TEMPLATE BASED ON THE criteria
}
}
感谢您提供的任何见解。
答案 0 :(得分:0)
我知道这是一个旧线程,但是最近我偶然发现了同样的问题。 如果其他人可以使用它,这是我为解决此问题所做的事情。
设置ContentControl的内容后,请执行以下操作:
Application.Current.Dispatcher.BeginInvoke(DispatcherPriority.Render, (Action)OnAfterRendered);
然后在“ OnAfterRendered”方法中,我可以将ContentControl与应用的DataTemplates一起使用