将控件保存为图片

原文:将控件保存为图片

                                        将控件保存为图片

                                                 周银辉

 

这里分别提供win form 和 wpf 两种方式

对于.net 2.0 我们可以简单地利用 BitBlt 函数来实现,非常简单,代码如下:

    public static class ControlToImageConverter
    {
        
private const Int32 SRCCOPY = 0xCC0020;

        [DllImport(
"gdi32.dll")]
        
internal static extern bool BitBlt(
            IntPtr hdcDest, 
// handle to destination DC
            int nXDest, // x-coord of destination upper-left corner
            int nYDest, // y-coord of destination upper-left corner
            int nWidth, // width of destination rectangle
            int nHeight, // height of destination rectangle
            IntPtr hdcSrc, // handle to source DC
            int nXSrc, // x-coordinate of source upper-left corner
            int nYSrc, // y-coordinate of source upper-left corner
            Int32 dwRop // raster operation code
            );


        
public static Bitmap GetImageOfControl(Control control)
        {

            var w 
= control.Size.Width;
            var h 
= control.Size.Height;

            Graphics gOfCtrl 
= control.CreateGraphics();
            var bmp 
= new Bitmap(w, h, gOfCtrl);

            Graphics gOfBmp 
= Graphics.FromImage(bmp);

            IntPtr dc1 
= gOfCtrl.GetHdc();
            IntPtr dc2 
= gOfBmp.GetHdc();

            BitBlt(dc2, 
00, w, h, dc1, 00, SRCCOPY);

            gOfCtrl.ReleaseHdc(dc1);
            gOfBmp.ReleaseHdc(dc2);

            gOfCtrl.Dispose();
            gOfBmp.Dispose();


            
return bmp;
        }

    }

 

其中 SRCCOPY = 0xCC0020, 表示将源直接拷贝到目标。

 

对于WPF,无法获得控件的hdc, 所以不能使用bitblt函数了,但幸运的是,其提供了一个RenderTargetBitmap的类型,可以将控件绘制到改类型的类型是示例上,比如:

 

var renderBitmap = 
         
new RenderTargetBitmap(w, h, 96d,96d, PixelFormats.Pbgra32);

renderBitmap.Render(controlToConvert);

 

其可以直接作为Image控件的源以便显示,如果要将其保存下来(保存到流中),则需要用到BitmapEncoder, 将上面的位图添加到encoder的帧中,然后调用save方法便可。

包装好的代码如下:

 

ExpandedBlockStart.gif点击展开
    internal enum ImageType
    {
        Bmp,
        Gif,
        Jpeg,
        Png,
        Tiff,
        Wdp
    }

    
internal static class ControlToImageConverter
    {

        
/// <summary>
        
/// Convert any control to a PngBitmapEncoder
        
/// </summary>
        
/// <param name="controlToConvert">The control to convert to an ImageSource</param>
        
/// <param name="imageType">The image type will indicate the type of return bitmap encoder</param>
        
/// <returns>The returned ImageSource of the controlToConvert</returns>
        private static BitmapEncoder GetImageFromControl(FrameworkElement controlToConvert, ImageType imageType)
        {

            var bounds 
= controlToConvert.GetBounds(controlToConvert.Parent as Visual);

            var renderBitmap 
= new RenderTargetBitmap((Int32)bounds.Width, (Int32)bounds.Height, 96d,
                                                      96d, PixelFormats.Pbgra32);
            renderBitmap.Render(controlToConvert);

            BitmapEncoder encoder 
= GetBitmapEncoderByImageType(imageType);

            
// puch rendered bitmap into it
            encoder.Frames.Add(BitmapFrame.Create(renderBitmap));

            
return encoder;
        }

        
private static Rect GetBounds(this FrameworkElement element, Visual from)
        {
            Rect rect 
= Rect.Empty;

            
try
            {
                GeneralTransform transform 
= element.TransformToVisual(from);
                rect 
= transform.TransformBounds(new Rect(00, element.ActualWidth, element.ActualHeight));
                
// ReSharper disable EmptyGeneralCatchClause
            }
            
catch
            
// ReSharper restore EmptyGeneralCatchClause
            {

            }

            
return rect;
        }

        
/// <summary>
        
/// Get an encoder by a specify image type
        
/// </summary>
        
/// <param name="type">the image type</param>
        
/// <returns>return an eccoder</returns>
        private static BitmapEncoder GetBitmapEncoderByImageType(ImageType type)
        {
            
switch (type)
            {
                
case ImageType.Bmp:
                    
return new BmpBitmapEncoder();
                
case ImageType.Gif:
                    
return new GifBitmapEncoder();
                
case ImageType.Jpeg:
                    
return new JpegBitmapEncoder();
                
case ImageType.Png:
                    
return new PngBitmapEncoder();
                
case ImageType.Tiff:
                    
return new TiffBitmapEncoder();
                
case ImageType.Wdp:
                    
return new WmpBitmapEncoder();
                
default:
                    
return new PngBitmapEncoder();
            }
        }

        
/// <summary>
        
/// Get the iamge type by image file name
        
/// </summary>
        
/// <param name="fileName">the file name of an image</param>
        
/// <returns>the iamge type</returns>
        private static ImageType GetImageTypeByFileName(string fileName)
        {
            ImageType returnType 
= ImageType.Png;

            var extension 
= Path.GetExtension(fileName);
            
if (!String.IsNullOrEmpty(extension))
            {
                
switch (extension.ToLower())
                {
                    
case ".bmp":
                        returnType 
= ImageType.Bmp;
                        
break;
                    
case ".gif":
                        returnType 
= ImageType.Gif;
                        
break;
                    
case ".jpeg":
                    
case ".jpg":
                    
case ".jpe":
                    
case "jfif":
                        returnType 
= ImageType.Jpeg;
                        
break;
                    
case ".png":
                        returnType 
= ImageType.Png;
                        
break;
                    
case ".tiff":
                    
case ".tif":
                        returnType 
= ImageType.Tiff;
                        
break;
                    
case ".wdp":
                        returnType 
= ImageType.Wdp;
                        
break;
                    
default:
                        returnType 
= ImageType.Png;
                        
break;
                }
            }

            
return returnType;
        }

        
/// <summary>
        
/// Get an ImageSource of a control
        
/// </summary>
        
/// <param name="controlToConvert">The control to convert to an ImageSource</param>
        
/// <param name="imageType">the image type</param>
        
/// <returns>The returned ImageSource of the controlToConvert</returns>
        public static BitmapSource GetImageOfControl(FrameworkElement controlToConvert, ImageType imageType)
        {
            
// return first frame of image 
            var encoder = GetImageFromControl(controlToConvert, imageType);
            
if (encoder != null && encoder.Frames != null && encoder.Frames.Count > 0)
            {
                
return encoder.Frames[0];
            }

            
return new BitmapImage();
        }

        
/// <summary>
        
/// Get an ImageSource of a control(Jpeg as default type)
        
/// </summary>
        
/// <param name="controlToConvert">The control to convert to an ImageSource</param>
        
/// <returns>The returned ImageSource of the controlToConvert</returns>
        public static BitmapSource GetImageOfControl(FrameworkElement controlToConvert)
        {
            
return GetImageOfControl(controlToConvert, ImageType.Jpeg);
        }

        
/// <summary>
        
/// Save an image of a control
        
/// </summary>
        
/// <param name="controlToConvert">The control to convert to an ImageSource</param>
        
/// <param name="fileName">The location to save the image to</param>
        
/// <returns>The returned ImageSource of the controlToConvert</returns>
        public static Boolean SaveImageOfControl(FrameworkElement controlToConvert, String fileName)
        {
            
try
            {
                var imageType 
= GetImageTypeByFileName(fileName);

                
using (var outStream = new FileStream(fileName, FileMode.Create))
                {
                    var encoder 
= GetImageFromControl(controlToConvert, imageType);
                    encoder.Save(outStream);
                }
            }
            
catch (Exception e)
            {
#if DEBUG
                Console.WriteLine(
"Exception caught saving stream: {0}", e.Message);
#endif
                
return false;
            }

            
return true;
        }

    }

 

 

注意到上面的GetImageOfControl方法返回的实际是BitmapFrame, 其是BitmapSource的一种,如果你更习惯使用BitmapImage的话,免费赠送如下方法:

        public static BitmapImage ToBitmapImage(this BitmapSource bitmapSource)
        {
            var encoder 
= new JpegBitmapEncoder();
            var memoryStream 
= new MemoryStream();

            encoder.Frames.Add(BitmapFrame.Create(bitmapSource));
            encoder.Save(memoryStream);

            var bmp 
= new BitmapImage();

            bmp.BeginInit();
            bmp.StreamSource 
= new MemoryStream(memoryStream.ToArray());
            bmp.EndInit();

            memoryStream.Flush();
            memoryStream.Close();

            
return bmp;
        }

 

 

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/277687.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Linux基础命令---get获取ftp文件

get 使用lftp登录ftp服务器之后&#xff0c;可以使用get指令从服务器获取文件。 1、语法 get [-E] [-a] [-c] [-O base] rfile [-o lfile] 2、选项列表 选项 说明 -o 指定输出文件的名字&#xff0c;不指定则使用原来的名字 -c 如果失败&#xff0c;持续获取 -E 获取…

Unity3D学习笔记之二资源导入以及工程管理

本次教程&#xff0c;我们来建立自己的一个工程并导入模型&#xff0c;对模型进行处理。特别说明&#xff1a;这系列教程参考自人人素材翻译组出品的翻译教程《Unity游戏引擎的基础入门视频教程》&#xff0c;本博客中采用的模型和方法均来源于此。这是我浏览过好多教程后觉得讲…

关于cocos creator换装功能的实践与思考

2019独角兽企业重金招聘Python工程师标准>>> 最近在做一个基于cocos creator的微信小游戏&#xff0c;其中一个主要的功能是给角色进行换装。先来说下开发环境&#xff1a; cocos creator1.9.3spine 在官方的文档中只有设置皮肤的接口&#xff0c;实际上是存在可以部…

pl/sql查询表数据,报错ORA-03115:不支持的网络数据类型或表示法

今天测试人员遇到一个问题&#xff1a;pl/sql查询表数据&#xff0c;报错ORA-03115&#xff1a;不支持的网络数据类型或表示法 我在plsql上查询这张表是没有问题的&#xff0c;去看了那张表&#xff0c;发现有个字段类型是binary_double&#xff0c; 查资料发现这是oracle10才出…

抖店一件代发实操,干货满满!

我是电商珠珠 没有货源的新手&#xff0c;在店铺刚开始的时候可以搞无货源模式&#xff0c;也就是一件代发&#xff0c;去搬运别人店铺的商品到自己店铺&#xff0c;再去利用信息差去赚取差价。 很多人不知道具体要怎么做&#xff0c;今天我就来给大家讲一讲。 一、入驻 入…

Unity3D学习笔记之三Prefab组件的使用

本次教程&#xff0c;我们来创建一个简单的Prefab组件。教程参考自人人素材翻译组出品的翻译教程《Unity游戏引擎的基础入门视频教程》。说到Prefab&#xff0c;中文翻译为预设体&#xff0c;在Unity官方的书本《Unity4.X从入门到精通》中的解释是&#xff1a;可以理解为是一个…

JavaScript之闭包

不少开发人员总是搞不清匿名函数和闭包这两个概念&#xff0c;因此经常混用。闭包是指有权访问另一个函数作用域中的变量的函数。创建闭包的常见方式&#xff0c;就是在一个函数内部创建另一个函数 --《JavaScript高级程序设计》 转载于:https://www.cnblogs.com/songsongblue/…

【跃迁之路】【522天】程序员高效学习方法论探索系列(实验阶段279-2018.07.12)...

(跃迁之路)专栏 【跃迁之路】奖励金计划正式开始 从2018.7.1起&#xff0c;【跃迁之路】奖励金计划正式起航&#xff0c;从今以后&#xff0c;每月1日&#xff0c;我会将自己个人上月收入的1%计入【跃迁之路】奖励金池&#xff0c;积累到足够金额后&#xff0c;将适时用于奖励那…

迷茫在路口——致我的2014

今天是2015年一月一日&#xff0c;2014年已经过去了。这意味着再过六个月整&#xff0c;我就要大学毕业&#xff0c;开始为自己的未来拼搏。这一年&#xff0c;我过的充实且迷茫。说2014充实&#xff0c;是因为我从寒假结束到有一个寒假的到来&#xff0c;几乎没有怎么闲下来&a…

Codeforces 173E Camping Groups 线段树

Camping Groups 我们先计算出&#xff0c; 每个点当leader所能掌控的最多人数。 然后我们把询问离线&#xff0c; 丢到responsibility最大的那个地方去。 然后从大到小往线段树里加人&#xff0c; 加入完之后处理掉当前的询问。 如果强制在线的话就只能树套树啦。 #include<…

tomcat闪退解决方案

在这几天&#xff0c;遇到一个Tomcat启动闪退的问题&#xff0c;通过查阅各种资料&#xff0c;算是完美解决。在此分享给朋友们。 首先&#xff0c;确定你的问题在哪里 1.查询错误&#xff1a;winR 输入cmd&#xff0c;进入一般处理程序。通过cd 找到你Tomcat的bin文件夹&#…

《古剑奇谭2》详细测评心得

期待已久的《古剑奇谭2》。仔仔细细的玩下来给我的感觉还是不错的。烛龙也不愧是国产单机的良心公司了&#xff0c;回合制的战斗方式改成了即时战斗类。 的确&#xff0c;国产动作类的游戏经验目前等于零。《古剑2》一改以往国产网游的作风跳出了回合制的圈子实属不易&#xff…

LeetCode 581. 最短无序连续子数组(Shortest Unsorted Continuous Subarray)

581. 最短无序连续子数组581. Shortest Unsorted Continuous Subarray 题目描述 给定一个整型数组&#xff0c;你需要寻找一个连续的子数组&#xff0c;如果对这个子数组进行升序排序&#xff0c;那么整个数组都会变为升序排序。 你找到的子数组应是最短的&#xff0c;请输出它…

NFS4文件锁机制探秘

2019独角兽企业重金招聘Python工程师标准>>> 简介 NFS4实现“租赁锁”。每个锁拥有一样的“租赁期”。客户端的读写操作将刷新“租赁期”。租赁期到期后&#xff0c;锁将被服务器释放。NFS4通过下述“模型”实现对锁的管理&#xff1a; 1) 清晰地划分客户端和服务器…

Stay Hungry Stay Foolish——网络学习平台分享

从1月24号回家也有一阵子了&#xff0c;今天已经是31号&#xff0c;这一个周的中心思想就是一个字&#xff0c;玩。 学生一但远离学校&#xff0c;就会碰到许多学习的阻力&#xff0c;有来自外界的&#xff0c;家里有活要干&#xff0c;有亲戚要访&#xff0c;有同学邀约&…

linux_check

linux_check echo "********CPU****************" echo 总核数 物理CPU个数 X 每颗物理CPU的核数 echo " 总逻辑CPU数 物理CPU个数 X 每颗物理CPU的核数 X 超线程数"echo 查看物理CPU个数 cat /proc/cpuinfo| grep "physical id"| sort| un…

Unity3D学习笔记之四完善Prefab并添加First Person Controller

好久没学东西并用博客记录了&#xff0c;这个年过的很懒散&#xff0c;慢慢临近开学了&#xff0c;也要提前适应一下&#xff0c;写写东西&#xff0c;这样开学才能更好的进入状态呀&#xff5e;&#xff5e;本次笔记中&#xff0c;我们将来雕琢一个更加完善的Prefab&#xff0…

高精度(压位+判负数+加减乘+读写)

本算法目前属于还处于测试状态&#xff0c;欢迎Hack&#xff01; struct gj{bool fu; //是否是负数int tt,mod; //高精的长度int s[40005]; //压位用的数组inline gj(){ //整体初始化fu0; tt0; mod1e9;memset(s,0,sizeof(s));}inline gj read(){ register char ch; //高精度读…

Hadoop从安装Linux到搭建集群环境

简介与环境准备  hadoop的核心是分布式文件系统HDFS以及批处理计算MapReduce。近年&#xff0c;随着大数据、云计算、物联网的兴起&#xff0c;也极大的吸引了我的兴趣&#xff0c;看了网上很多文章&#xff0c;感觉还是云里雾里&#xff0c;很多不必要的配置都在入门教程出现…

git推送本地分支到远程分支

场景 有时候我们开发需要开一个分支,这样可以有效的并行开发. 开分支有两种方式: 一种是在远程开好分支,本地直接拉下来;一种是本地开好分支,推送到远程.远程先开好分支然后拉到本地 git checkout -b feature-branch origin/feature-branch //检出远程的feature-branch分支到…