3

我的印象是,听Loaded 事件将确保我可以在那时访问可视化树。这似乎在大多数情况下都可以正常工作,而可视化树正是我所期望的。但是,当使用TabControl时,隐藏选项卡上的控件不会在加载的事件触发之前构建其可视化树。

举个简单的例子:

<Window ...>
    <Grid>
        <TabControl>
            <TabItem Header="Tab 1">
                <my:MyTextBox />
            </TabItem>
            <TabItem Header="Tab 2">
                <my:MyTextBox />
            </TabItem>
        </TabControl>
    </Grid>
</Window>

MyTextBox是一个自定义组件,仅在发生某些事件时注销。

public class MyTextBox : TextBox
{
    public MyTextBox()
    {
        Loaded += (s,e) => {
            Console.WriteLine("MyTextBox {0}: Loaded", s.GetHashCode());
            Dispatcher.BeginInvoke(()=>Console.WriteLine("MyTextBox {0}: Loaded BeginInvoke", s.GetHashCode()));
        };
    }

    public override void OnApplyTemplate()
    {
        base.OnApplyTemplate();
        Console.WriteLine("MyTextBox {0}: OnApplyTemplate", this.GetHashCode());
    }
}

这是输出:

MyTextBox A: OnApplyTemplate
MyTextBox A: Loaded
MyTextBox B: Loaded
MyTextBox A: Loaded BeginInvoke
MyTextBox B: Loaded BeginInvoke
(changing tab to 2)
MyTextBox B: OnApplyTemplate
MyTextBox B: Loaded
MyTextBox B: Loaded BeginInvoke

如您所见,B第二个选项卡上的forOnApplyTemplate仅在您切换到其选项卡时才会执行,但加载的事件实际上是在之前引发的。

现在的问题是我的自定义组件是一个没有外观的组件,它设置了控制模板。在加载的事件中,我现在想要浏览可视化树并应用一些额外的默认值,但由于尚未应用模板,因此可视化树是空的。

是否有某种方法可以从外部(即不修改组件)确保我可以在加载可视树后立即对其进行迭代?

4

1 回答 1

0

The TabControl does some virtualization under the hood. You can disable this as per answers in this SO question: How to preserve control state within tab items in a TabControl

于 2012-09-27T12:10:36.960 回答