在TabControl的另一个选项卡中呈现内容控件

时间:2010-01-09 05:06:33

标签: c# wpf tabcontrol render contentcontrol

我有一个程序,其中有多个选项卡以编程方式动态添加到TabControl对象。我想要做的是将每个选项卡的内容值呈现为PNG。我正在使用我在StackOverflow上的其他地方或者谷歌上找到的脚本(丢失了源代码)。我的代码如下所示:

if (tabPanel.Items.Count > 0)
{
    SaveFileDialog fileDialog = new SaveFileDialog();
    fileDialog.Filter = "PNG|*.png";
    fileDialog.Title = "Save Tabs";
    fileDialog.ShowDialog();

    if (fileDialog.FileName.Trim().Length > 0)
    {
        try
        {
            string filePrefix = fileDialog.FileName.Replace(".png", "");
            int tabNo = 1;
            foreach (TabItem tabItem in tabPanel.Items)
            {
                string filename = filePrefix + "_" + tabNo + ".png";

                TabContentControl content = tabItem.Content as TabContentControl;
                Rect rect = new Rect(content.RenderSize);
                RenderTargetBitmap rtb = new RenderTargetBitmap((int)rect.Right, (int)rect.Bottom, 96d, 96d, System.Windows.Media.PixelFormats.Default);
                rtb.Render(content);

                BitmapEncoder pngEncoder = new PngBitmapEncoder();
                pngEncoder.Frames.Add(BitmapFrame.Create(rtb));

                System.IO.MemoryStream ms = new System.IO.MemoryStream();
                pngEncoder.Save(ms);
                System.IO.File.WriteAllBytes(filename, ms.ToArray());
                ms.Close();

                tabNo++;
            }
        }
        catch (Exception ex)
        {
            // log exception
        }
    }
}

此代码按照需要运行如果我已经查看并查看了在调用此代码之前必须呈现的所有选项卡。它继续并使用从TabContentControl呈现的正确内容创建filePrefix_1.png,filePrefix_2.png等。但是,如果我在查看所有选项卡之前调用使用此代码的处理程序,则我的代码会在new RenderTargetBitmap(...)处抛出异常,因为content.RenderSize{0.0, 0.0}。当我尝试将未查看的标签的渲染大小强制为一次查看时,我输出的PNG尺寸正确但完全为空。

所以我想我需要一些方法来强制渲染TabContentControl。看起来像Render事件只在需要渲染UIElement时运行。他们能解决这个问题吗?

我还试图通过在Page_Loaded事件处理程序中创建选项卡时添加以下代码来“欺骗”WPF绘制选项卡内容:

void Page_Loaded(object sender, RoutedEventArgs e)
{
    // irrelevant code
    foreach (// iterate over content that is added to each tab)
    {
        TabItem tabItem = new TabItem();
        // load content
        tabPanel.Items.Add(tabItem);
        tabItem.IsSelected = true;
    }
    // tabPanel.SelectedIndex = 0;
}

Page_Loaded处理程序中的最后一行被注释掉时,最后一个选项卡处于焦点位置,并为其内容定义了RenderSize属性。如果未注释掉最后一行,则第一个选项卡处于焦点,具有相同的行为。其他选项卡没有任何渲染信息。

2 个答案:

答案 0 :(得分:3)

最后想通了,感谢this blog post。解决方案涉及使用Refresh方法为UIElement创建扩展方法,该方法调用具有渲染优先级的空委托。显然,使用渲染优先级调度某些内容会导致执行所有其他更重要的项目,从而更改选项卡。

此处复制代码,以防博客被删除:

public static class ExtensionMethods
{
   private static Action EmptyDelegate = delegate() { };

   public static void Refresh(this UIElement uiElement)
   {
      uiElement.Dispatcher.Invoke(DispatcherPriority.Render, EmptyDelegate);
   }
}

void Page_Loaded(object sender, RoutedEventArgs e)
{
    // irrelevant code
    foreach (// iterate over content that is added to each tab)
    {
        TabItem tabItem = new TabItem();
        // load content
        tabPanel.Items.Add(tabItem);
        tabItem.IsSelected = true;
        tabItem.Refresh();
    }
    // tabPanel.SelectedIndex = 0;
}

要使用它,只需在您需要使用此功能的代码文件中包含扩展名称空间,它将出现在方法列表中。

答案 1 :(得分:1)

这不是一个理想的解决方案,但是当您创建选项卡并将其添加到选项卡控件时,您只需存储当前打开的选项卡,然后切换到刚刚创建的选项卡,然后切换回。对于用户来说,它只是轻微的闪烁,但实际上你只是欺骗了winforms(或相应的wpf)来绘制对象。