如何在WPF中制作UIElement的屏幕截图

2024-02-03

我在创建 scatterview 的屏幕截图时遇到问题。我的屏幕截图总是包含黑框。

这是我的 XAML 代码:

<s:SurfaceWindow
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:s="http://schemas.microsoft.com/surface/2008"
    xmlns:d="http://schemas.microsoft.com/expression/blend/2008" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" mc:Ignorable="d" x:Class="MakeScreenshots.SurfaceWindow1"
    Title="MakeScreenshots" Width="1000" Height="700"
    >
  <s:SurfaceWindow.Resources>
    <ImageBrush x:Key="WindowBackground" Stretch="None" Opacity="0.6" ImageSource="pack://application:,,,/Resources/WindowBackground.jpg"/>
  </s:SurfaceWindow.Resources>

  <Grid x:Name="GlobalGrid" Background="{StaticResource WindowBackground}" Width="1000" Height="700" >

    <s:ScatterView x:Name="ScatterViewScreenShot" Margin="108,89,176,73" Width="700" Height="500">
        <s:ScatterView.Background>
            <SolidColorBrush Color="{DynamicResource {x:Static SystemColors.ActiveCaptionColorKey}}"/>
        </s:ScatterView.Background>
        <s:ScatterViewItem Margin="0,-26.953,-130.946,-23.047" Content="ScatterViewItem 3" HorizontalAlignment="Right" Width="125.826"/>
        <s:ScatterViewItem Margin="0,0,-490.513,-151.256" HorizontalAlignment="Right" Width="125.77" Height="60.427" VerticalAlignment="Bottom" Content="ScatterViewItem 2"/>
        <s:ScatterViewItem Content="ScatterViewItem 1" Margin="-331.43,0,0,-129.589" HorizontalAlignment="Left" Width="177.949" Height="67.905" VerticalAlignment="Bottom"/>
    </s:ScatterView>
    <Button x:Name="MakeScreenShotButton" Click="MakeScreenShotButton_Click" Content="MakeScreenShot" Margin="267,17,343,0" VerticalAlignment="Top" Height="38.96"/>
        <Button Content="MakeScreenShotButton2" Height="39" HorizontalAlignment="Left" Margin="116,614,0,0" Name="button1" VerticalAlignment="Top" Width="301" Click="MakeScreenShotButton2_Click" />
        <Button Content="MakeScreenShotButton3"  Click="MakeScreenShotButton3_Click" Height="39" HorizontalAlignment="Left" Margin="822,207,0,0" Name="button2" VerticalAlignment="Top" Width="147" />
        <Button Content="MakeScreenShotButton4" Click="MakeScreenShotButton4_Click" Height="39" HorizontalAlignment="Left" Margin="822,349,0,0" Name="button3" VerticalAlignment="Top" Width="147" />
        <Button Content="MakeScreenShotButton5" Click="MakeScreenShotButton5_Click" Height="39" HorizontalAlignment="Left" Margin="822,443,0,0" Name="button4" VerticalAlignment="Top" Width="147" />
    </Grid>
</s:SurfaceWindow>

这是 C# 代码:

    private void MakeScreenShotButton_Click(object sender, RoutedEventArgs e)
    {          
        RenderTargetBitmap targetBitmap = new RenderTargetBitmap((int)ScatterViewScreenShot.ActualWidth, (int)ScatterViewScreenShot.ActualHeight, 80d, 80d, PixelFormats.Default);

        targetBitmap.Render(ScatterViewScreenShot);

        // add the RenderTargetBitmap to a Bitmapencoder

        JpegBitmapEncoder encoder = new JpegBitmapEncoder();
        encoder.Frames.Add(BitmapFrame.Create(targetBitmap));

        // Encoder zum Speichern des Bildes
        JpegBitmapEncoder encoderToSave = new JpegBitmapEncoder();
        encoderToSave.Frames.Add(BitmapFrame.Create(targetBitmap));

        // Speichern des Bildes auf der Festplatte
        string fileName = "M:\\TestForStackOverflow.jpg";
        System.IO.FileStream fs = System.IO.File.Open(fileName, System.IO.FileMode.OpenOrCreate);
        encoderToSave.Save(fs);

        encoder.QualityLevel = 40;

        MemoryStream ms = new MemoryStream();
        encoder.Save(ms);

        // Convert Image to byte[]
        byte[] imageBytes = ms.ToArray();

        int anzahlBytes = imageBytes.Length;

        string imageAsBase64String = Convert.ToBase64String(imageBytes);

        TCP_Client client = new TCP_Client("192.168.5.3", 4321);
        client.sendeNachricht(imageAsBase64String);
    }

    private void MakeScreenShotButton2_Click(object sender, RoutedEventArgs e)
    {
        // 1. Bitmap der gewünschten Größe erstellen
        int width = (int)ScatterViewScreenShot.ActualWidth;
        int height = (int)ScatterViewScreenShot.ActualHeight;
        RenderTargetBitmap rtb = new RenderTargetBitmap(width, height, 100d, 100d, PixelFormats.Default); // 500 x 500 genau wie bei den Produktfotos

        // 2. Control in Bitmap hinein rendern
        Visual vis = (Visual)ScatterViewScreenShot;
        rtb.Render(vis);

        // 3. Control-Image erzeugen und dem Control-Image als Source das Bitmap übergeben
        System.Windows.Controls.Image img = new System.Windows.Controls.Image();
        img.Source = rtb;
        img.Stretch = Stretch.None;

        // 4. Aktualisieren der Größe des Elements entsprechend des Inhaltes mittels der Methoden Measure und Arrange.
        img.Measure(new System.Windows.Size(width, height));
        System.Windows.Size sizeImage = img.DesiredSize;
        img.Arrange(new System.Windows.Rect(new System.Windows.Point(0, 0), sizeImage));

        // 5. Image wird mit der korrekten Größe erneut gerendert und an PngBitmapEncoder übergeben
        RenderTargetBitmap rtb2 = new RenderTargetBitmap((int)rtb.Width, (int)rtb.Height, 60, 60, PixelFormats.Default);
        rtb2.Render(img);

        PngBitmapEncoder jpeg = new PngBitmapEncoder();
        jpeg.Frames.Add(BitmapFrame.Create(rtb2));


        // 6. Image in Stream schreiben
        MemoryStream ms = new MemoryStream();
        jpeg.Save(ms);

        // Convert Image to byte[]
        byte[] imageBytes = ms.ToArray();

        int anzahlBytes = imageBytes.Length;

        string imageAsBase64String = Convert.ToBase64String(imageBytes);

        TCP_Client client = new TCP_Client("192.168.5.3", 4321);
        client.sendeNachricht(imageAsBase64String);
    }

    private void MakeScreenShotButton3_Click(object sender, RoutedEventArgs e)
    {
        int width_x = 240;
        int width_y = 400;

        Bitmap screen = TakeScreenshot(100, 100, width_x, width_y);

        System.Drawing.Image img = (System.Drawing.Image)screen;
        //Image img = Image.FromFile("bla.jpg");
        MemoryStream ms = new MemoryStream();

        try
        {
            //Ein ImageCodecInfo-Objekt für den JPEG-Codec anlegen
            ImageCodecInfo jpegCodec = null;

            //Den Qualitätsarameter konfigurieren (Qualitätsfaktor in
            //Prozent angeben)

            EncoderParameter qualitaetsParameter = new EncoderParameter(
                        System.Drawing.Imaging.Encoder.Quality, 40);

            //Alle im System verfügbaren Codecs auflisten
            ImageCodecInfo[] alleCodecs = ImageCodecInfo.GetImageEncoders();

            EncoderParameters codecParameter = new EncoderParameters(1);
            codecParameter.Param[0] = qualitaetsParameter;

            //Den JPEG-Codec unter allen Codecs finden und dem
            //Codec-Info-Objekt zuweisen
            for (int i = 0; i < alleCodecs.Length; i++)
            {
                if (alleCodecs[i].MimeType == "image/jpeg")
                {
                    jpegCodec = alleCodecs[i];
                    break;
                }
            }

            // Bild in Stream schreiben
            img.Save(ms, jpegCodec, codecParameter);
        }
        catch (ArgumentException w)
        {
            throw w;
        }

        //  Console.WriteLine(StreamToBase64(ms, System.Drawing.Imaging.ImageFormat.Jpeg));
        String bildBase64 = StreamToBase64(ms, System.Drawing.Imaging.ImageFormat.Jpeg);

        TCP_Client client = new TCP_Client("192.168.5.3", 4321);
        client.sendeNachricht(bildBase64);
    }

    private void MakeScreenShotButton4_Click(object sender, RoutedEventArgs e)
    {
        // save current canvas transform
        Transform transform = ScatterViewScreenShot.LayoutTransform;

        // get size of control
        System.Windows.Size sizeOfControl = new System.Windows.Size(ScatterViewScreenShot.ActualWidth, ScatterViewScreenShot.ActualHeight);
        // measure and arrange the control
        ScatterViewScreenShot.Measure(sizeOfControl);
        // arrange the surface
        ScatterViewScreenShot.Arrange(new Rect(sizeOfControl));

        // craete and render surface and push bitmap to it
        RenderTargetBitmap renderBitmap = new RenderTargetBitmap((Int32)sizeOfControl.Width, (Int32)sizeOfControl.Height, 96d, 96d, PixelFormats.Pbgra32);
        // now render surface to bitmap
        renderBitmap.Render(ScatterViewScreenShot);

        // encode png data
        PngBitmapEncoder pngEncoder = new PngBitmapEncoder();
        // puch rendered bitmap into it
        pngEncoder.Frames.Add(BitmapFrame.Create(renderBitmap));

        /* Speichern des Bildes auf der Festplatte
        string fileName = "M:\\ScreenshotClick4.jpg";
        System.IO.FileStream fs = System.IO.File.Open(fileName, System.IO.FileMode.OpenOrCreate);
        pngEncoder.Save(fs);*/

        // Encoder zum Senden des Bildes
        JpegBitmapEncoder encoder = new JpegBitmapEncoder();
        encoder.Frames.Add(BitmapFrame.Create(renderBitmap));

        MemoryStream ms = new MemoryStream();
        encoder.Save(ms);

        // Convert Image to byte[]
        byte[] imageBytes = ms.ToArray();

        int anzahlBytes = imageBytes.Length;

        string imageAsBase64String = Convert.ToBase64String(imageBytes);

        TCP_Client client = new TCP_Client("192.168.5.3", 4321);
        client.sendeNachricht(imageAsBase64String);

    }

    private void MakeScreenShotButton5_Click(object sender, RoutedEventArgs e)
    {
        int scale = 1;

        double actualHeight = ScatterViewScreenShot.RenderSize.Height;
        double actualWidth = ScatterViewScreenShot.RenderSize.Width;

        double renderHeight = actualHeight * scale;
        double renderWidth = actualWidth * scale;

        RenderTargetBitmap renderTarget = new RenderTargetBitmap((int)renderWidth, (int)renderHeight, 96, 96, PixelFormats.Pbgra32);
        VisualBrush sourceBrush = new VisualBrush(ScatterViewScreenShot);

        DrawingVisual drawingVisual = new DrawingVisual();
        DrawingContext drawingContext = drawingVisual.RenderOpen();

        using (drawingContext)
        {
            drawingContext.PushTransform(new ScaleTransform(actualWidth, actualHeight));
            drawingContext.DrawRectangle(sourceBrush, null, new Rect(new System.Windows.Point(0, 0), new System.Windows.Point(actualWidth, actualHeight)));
        }
        renderTarget.Render(drawingVisual);

        JpegBitmapEncoder jpgEncoder = new JpegBitmapEncoder();
        jpgEncoder.QualityLevel = 40;
        jpgEncoder.Frames.Add(BitmapFrame.Create(renderTarget));

        MemoryStream ms = new MemoryStream();
        jpgEncoder.Save(ms);

        // Convert Image to byte[]
        byte[] imageBytes = ms.ToArray();

        int anzahlBytes = imageBytes.Length;

        string imageAsBase64String = Convert.ToBase64String(imageBytes);

        TCP_Client client = new TCP_Client("192.168.5.3", 4321);
        client.sendeNachricht(imageAsBase64String);

    }

    #region Hilfsmethoden

    public string StreamToBase64(MemoryStream ms, System.Drawing.Imaging.ImageFormat format)
    {
        // Convert Image to byte[]
        byte[] imageBytes = ms.ToArray();

        // Convert byte[] to Base64 String
        string base64String = Convert.ToBase64String(imageBytes);
        return base64String;
    }

    private Bitmap TakeScreenshot(int StartX, int StartY, int Width, int Height)
    {
        Bitmap Screenshot = new Bitmap(Width, Height);
        Graphics G = Graphics.FromImage(Screenshot);

        G.CopyFromScreen(StartX, StartY, 0, 0, new System.Drawing.Size(Width, Height), CopyPixelOperation.SourceCopy);
        return Screenshot;
    }

    #endregion


}
}

我发现以下文章可以解决您的问题: Take WPF “Screenshots” (Link https://www.grumpydev.com/2009/01/03/taking-wpf-screenshots/到博客文章)

///
/// Gets a JPG "screenshot" of the current UIElement
///
/// UIElement to screenshot
/// Scale to render the screenshot
/// JPG Quality
/// Byte array of JPG data
public static byte[] GetJpgImage(this UIElement source, double scale, int quality)
{
    double actualHeight = source.RenderSize.Height;
    double actualWidth = source.RenderSize.Width;

    double renderHeight = actualHeight * scale;
    double renderWidth = actualWidth * scale;

    RenderTargetBitmap renderTarget = new RenderTargetBitmap((int) renderWidth, (int) renderHeight, 96, 96, PixelFormats.Pbgra32);
    VisualBrush sourceBrush = new VisualBrush(source);

    DrawingVisual drawingVisual = new DrawingVisual();
    DrawingContext drawingContext = drawingVisual.RenderOpen();

    using (drawingContext)
    {
        drawingContext.PushTransform(new ScaleTransform(scale, scale));
        drawingContext.DrawRectangle(sourceBrush, null, new Rect(new Point(0, 0), new Point(actualWidth, actualHeight)));
    }
    renderTarget.Render(drawingVisual);

    JpegBitmapEncoder jpgEncoder = new JpegBitmapEncoder();
    jpgEncoder.QualityLevel = quality;
    jpgEncoder.Frames.Add(BitmapFrame.Create(renderTarget));

    Byte[] _imageArray;

    using (MemoryStream outputStream = new MemoryStream())
    {
        jpgEncoder.Save(outputStream);
        _imageArray = outputStream.ToArray();
    }

    return _imageArray;
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

如何在WPF中制作UIElement的屏幕截图 的相关文章

随机推荐

  • AntiForgeryToken 过期空白页

    我将 IdentityServer4 与 ASP NET Core 2 2 一起使用 在 登录后 方法中 我应用了 ValidateAntiForgeryToken 通常 在登录页面停留 20 分钟到 2 小时后尝试登录 会出现一个空白页面
  • 如何编写递归函数来反转链表?

    我想用 Python 来做这件事 我不想只是反向打印它 而是实际上反转给定的节点 我见过其他语言的实现 但在 Python 中找不到示例 我试图在一个函数中完成它 但如果需要辅助函数 那就这样吧 def reverse item tail
  • 更新嵌套对象 firebase

    来自 Firebase 注释 给定一个单一的关键路径 例如alanisawesome updateChildren 仅更新第一个子级别的数据 并且超出第一个子级别传入的任何数据都将被视为setValue 手术 多路径行为允许更长的路径 例如
  • 如何在Flutter中获取当前时区区域(tz数据库名称)

    我是 flutter 新手 我的其余 api 将当前时区设置为欧洲 伦敦 我不知道如何在颤振中获取当前时区 stackoverflow 上有一个关于这个主题的问题 但没有用户给出答案 Flutter 时区 作为 ZoneId https s
  • Foundation 5 网格推拉

    我正在使用 Foundation 5 并尝试在移动屏幕上实现以下 DIV 布局 A B C D E 我希望在桌面屏幕上显示如下
  • 使用 AndroidPlot 自定义图表上的点

    任何使用过 AndroidPlot 库的人都会告诉我如何在图表上绘制自定义点 到目前为止 我正在使用 LineAndPointRenderer 类并将线条设置为透明 我想至少改变点的大小 但如果可能的话有一个自定义图像 P S 拥有 150
  • JavaScript - 测试整数

    我有一个文本字段 允许用户输入他们的年龄 我正在尝试使用 JavaScript 对此字段进行一些客户端验证 我已经有了服务器端验证 但是 我似乎无法验证用户是否输入了实际的整数 我目前正在尝试以下代码 function IsValidAge
  • 编辑单元格时刷新行

    我对这个 JTable 有问题 我像这样编辑一个单元格 然后我按 Enter 键提交更改 在这里 我希望表 GUI 能够刷新新值 但它们没有显示 只有当我像这样更改选择时才会显示 fireTableCellUpdated inRow inC
  • 用于运行任务名称中包含连字符的 Ant 任务的命令行

    任务名称以连字符 开头
  • 如何在同一个按钮上使用反应路由器和反应滚动?

    我想在同一个导航链接上使用反应路由器和反应滚动 如果我在主页上 我希望按钮具有反应滚动行为 如果我在不同的页面上 按钮需要返回主页 然后滚动到首选组件 反应滚动行为 这是我的想法 但它不能正常工作 import Link from reac
  • 使用 kubernetes nginx-ingress 反向代理具有 SNI 支持的站点

    我正在使用 kubernetes nginx ingress 设置反向代理 但我不知道如何将 nginx 参数添加到配置中 具体来说 proxy ssl 服务器名称 http nginx org en docs http ngx http
  • 如何在其他代码行同时执行的同时播放声音?

    我希望我的代码能够执行此操作 但要在后台播放音乐 import time while True print ligma time sleep 1 5 我试过这个 import time import winsound while True
  • 为什么我的数组加载后会清空?

    我一定错过了一些基本的东西 但我似乎无法保持我的数组加载 它加载成功 但当它出现在另一个函数中时 它变成空的 我的目标是从位置数组中随机选择一个城市 地图视图加载了我的所有注释 但是当我在注释出现后调用 pickRandomNumber 时
  • 无法连接到 Poloniex

    我正在尝试连接到 Poloniex 为了进行连接 我使用 WynthShop 和以下代码 public class Program public static void Main string args var channelFactory
  • Oracle Database12c ORA 01918 和连接错误

    我已经安装了 Oracle 12c 但在创建和使用第一个数据库时遇到问题 我运行 SQL Developer 并使用 hr 用户 但它一直告诉我该帐户已锁定 我搜索了 stackoverflow 答案和官方文档 并尝试使用以下方法解锁它 A
  • WPF ShowDialog 在第二次调用时立即返回 null

    我认为这是 WPF 框架中的一个错误 没有深入了解我的程序以及为什么我要做我正在做的事情 我编写了一个简单的测试应用程序来证明我的理论 这个问题能否得到证实 在将应用程序放入其运行循环之前执行一系列对话框有哪些可能的解决方法 using S
  • 如何在 Modelica 中构建液体平衡连接器?

    帖子状态 200313 得到了代码 DEMO v42 的答案 我接受了赏金 200310 我评论了昨天建议的两篇关键论文 还是不明白如何更新DEMO v41 200309 我想强调的是 关键问题是如何在代码DEMO v41 如果可能的话 中
  • CodeIgniter框架中的库和模型有什么区别?

    我是 CI 新手 之前我曾在 RoR 和 Symfony 中广泛工作过 我无法理解为什么 CI 提供了库和模型 我检查的大多数 CI 代码通常使用库作为模型的包装器 模型专门用于与数据库对话 有人可以阐明这一点吗 对此可能还有其他的思想流派
  • Django:精确查找的 QuerySet 值必须使用切片限制为一个结果

    我正在开发一个项目 管理员可以将团队分配给经理 但它不起作用 我不知道它将如何工作 因为它会引发一个错误 指出 精确查找的 QuerySet 值必须使用切片限制为一个结果 这是我的 model py class manager models
  • 如何在WPF中制作UIElement的屏幕截图

    我在创建 scatterview 的屏幕截图时遇到问题 我的屏幕截图总是包含黑框 这是我的 XAML 代码