本文将告诉大家如何在 WPF 里面设置图片的 EXIF 信息,包括如何设置图片的旋转信息,以及如何读取 EXIF 的内容

值得一提的是在 WPF 里面,默认的图片渲染信息是无视 System.Photo.Orientation 信息的,一切都是推荐进行手动控制

在开始之前,咱先来用代码创建一张简单的图片

在 WPF 里面,使用代码进行绘图是一个非常高性能的方法,可以重复使用 DirectX 提供的高性能绘制能力,再加上 WPF 特别友好的 API 进行绘制。而且 WPF 的上层 API 是统一的,屏蔽掉很多细节,不需要更多额外的知识即可使用

先创建一个 DrawingVisual 对象,在这里面传入想要绘制的内容,接着使用 RenderTargetBitmap 存放为图片,最后再使用编码器生成图片

以下是创建 DrawingVisual 进行简单绘制的代码

        var drawingVisual = new DrawingVisual();
using (var drawingContext = drawingVisual.RenderOpen())
{
drawingContext.DrawRectangle(Brushes.White, null, new Rect(new Size(100, 10)));
drawingContext.DrawLine(new Pen(Brushes.Black, 2), new Point(2, 5), new Point(90, 5));
}

这里先绘制一个矩形是为了撑开范围,作为画布大小

以上代码准确来说,是没有进行任何实际的绘制逻辑,只是告诉 WPF 框架,应该如何进行绘制。那啥时候才能进行绘制?需要在实际用到绘制的时候,才会调用 DirectX 进行绘制,例如调用 RenderTargetBitmap 的 Render 方法,将绘制的指令转换为绘制为内存的图片

        var dpiScale = VisualTreeHelper.GetDpi(this);
var renderTargetBitmap = new RenderTargetBitmap(100, 10, dpiScale.PixelsPerInchX, dpiScale.PixelsPerInchY,
PixelFormats.Pbgra32);
renderTargetBitmap.Render(drawingVisual);

想要将内存里的图片存放到文件里面,就需要先对此按照一定的规则进行编码,例如本文将使用 Jpg 编码

先新建编码器

        var jpegBitmapEncoder = new JpegBitmapEncoder();

接着传入 RenderTargetBitmap 用来编码

        var bitmapFrame = BitmapFrame.Create(renderTargetBitmap);
jpegBitmapEncoder.Frames.Add(bitmapFrame);

最后输出为文件

        var file = Path.GetTempFileName() + ".jpg";
using (var fileStream = new FileStream(file, FileMode.Create, FileAccess.ReadWrite))
{
jpegBitmapEncoder.Save(fileStream);
}

在 BitmapFrame 的创建代码里,可以指定 BitmapMetadata 用来设置 EXIF 信息,例如创建了以下的 BitmapMetadata 对象

        var bitmapMetadata = new BitmapMetadata("jpg")
{
Title = "旋转的图片",
Author = new ReadOnlyCollection<string>(new[] { "林德熙" }),
Comment = "这是备注",
Copyright = "版权",
Subject = "主题",
ApplicationName = "应用",
};

那么从保存的图片的属性即可拿到上面传入的内容

本文所需要的旋转信息,也是设置到 BitmapMetadata 内容,使用 SetQuery 方法进行设置

        const int Rotate90 = 6;
bitmapMetadata.SetQuery("System.Photo.Orientation", Rotate90);

除了这个旋转信息,能写入属性有哪些,写入的类型是什么?这些可以从 官方文档 获取

以下就是全部的创建图片的代码

        var drawingVisual = new DrawingVisual();
using (var drawingContext = drawingVisual.RenderOpen())
{
drawingContext.DrawRectangle(Brushes.White, null, new Rect(new Size(100, 10)));
drawingContext.DrawLine(new Pen(Brushes.Black, 2), new Point(2, 5), new Point(90, 5));
} var dpiScale = VisualTreeHelper.GetDpi(this);
var renderTargetBitmap = new RenderTargetBitmap(100, 10, dpiScale.PixelsPerInchX, dpiScale.PixelsPerInchY,
PixelFormats.Pbgra32);
renderTargetBitmap.Render(drawingVisual); var jpegBitmapEncoder = new JpegBitmapEncoder();
var bitmapMetadata = new BitmapMetadata("jpg")
{
Title = "旋转的图片",
Author = new ReadOnlyCollection<string>(new[] { "林德熙" }),
Comment = "这是备注",
Copyright = "版权",
Subject = "主题",
ApplicationName = "应用",
};
const int Rotate90 = 6;
bitmapMetadata.SetQuery("System.Photo.Orientation", Rotate90); var bitmapFrame = BitmapFrame.Create(renderTargetBitmap, thumbnail: null, bitmapMetadata,
new ReadOnlyCollection<ColorContext>(new List<ColorContext>()));
jpegBitmapEncoder.Frames.Add(bitmapFrame); var file = Path.GetTempFileName() + ".jpg";
using (var fileStream = new FileStream(file, FileMode.Create, FileAccess.ReadWrite))
{
jpegBitmapEncoder.Save(fileStream);
}

接下来在 XAML 放一个 Image 控件,设置图片文件给图片控件,可以看到图片控件是无视 System.Photo.Orientation 的内容

        Image.Source = new BitmapImage(new Uri(file));

接下来尝试读取解析图片的 EXIF 信息,获取到旋转角度

先进行解码,从本地文件获取到图片文件,需要将图片文件解码,才能在内存里面认识这个图片

        var decoder = BitmapDecoder.Create
(
new Uri(file, UriKind.Absolute),
BitmapCreateOptions.DelayCreation,
BitmapCacheOption.None
);

为了性能考虑,这里只是读取 EXIF 信息,就设置了 DelayCreation 标记,如此就不会全部解析图片的内容

获取 BitmapFrame 内容

        var frame = decoder.Frames[0];

接着读取 BitmapMetadata 信息

        const string query = "System.Photo.Orientation";
return frame.Metadata is BitmapMetadata bitmapMetadata
&& bitmapMetadata.ContainsQuery(query)
&& bitmapMetadata.GetQuery(query) is ushort orientation
? orientation switch
{
6 => Rotation.Rotate90,
3 => Rotation.Rotate180,
8 => Rotation.Rotate270,
_ => Rotation.Rotate0,
}
: Rotation.Rotate0;

根据旋转角度,可以获取到显示的尺寸

   var size = rotation is Rotation.Rotate90 or Rotation.Rotate270
? new Size(frame.PixelHeight, frame.PixelWidth)
: new Size(frame.PixelWidth, frame.PixelHeight);

全部的读取代码如下

    private void Decode(string file)
{
var decoder = BitmapDecoder.Create
(
new Uri(file, UriKind.Absolute),
BitmapCreateOptions.DelayCreation,
BitmapCacheOption.None
); var frame = decoder.Frames[0]; var rotation = GetRotation(frame); var size = rotation is Rotation.Rotate90 or Rotation.Rotate270
? new Size(frame.PixelHeight, frame.PixelWidth)
: new Size(frame.PixelWidth, frame.PixelHeight);
} private Rotation GetRotation(BitmapFrame frame)
{
const string query = "System.Photo.Orientation";
return frame.Metadata is BitmapMetadata bitmapMetadata
&& bitmapMetadata.ContainsQuery(query)
&& bitmapMetadata.GetQuery(query) is ushort orientation
? orientation switch
{
6 => Rotation.Rotate90,
3 => Rotation.Rotate180,
8 => Rotation.Rotate270,
_ => Rotation.Rotate0,
}
: Rotation.Rotate0;
}

本文代码放在githubgitee 欢迎访问

可以通过如下方式获取本文的源代码,先创建一个空文件夹,接着使用命令行 cd 命令进入此空文件夹,在命令行里面输入以下代码,即可获取到本文的代码

git init
git remote add origin https://gitee.com/lindexi/lindexi_gd.git
git pull origin aee84bc738984fe0df51bea703ace2e721f9ec99

以上使用的是 gitee 的源,如果 gitee 不能访问,请替换为 github 的源。请在命令行继续输入以下代码

git remote remove origin
git remote add origin https://github.com/lindexi/lindexi_gd.git
git pull origin aee84bc738984fe0df51bea703ace2e721f9ec99

获取代码之后,进入 HurcadahejaiYawhekurji 文件夹

在 UWP 里,写入 EXIF 信息,请参阅 UWP 写入图片 Exif 信息

WPF 通过 EXIF 设置和读取图片的旋转信息的更多相关文章

  1. WPF Image Source 设置相对路径图片

    BitmapImage bt = new BitmapImage(new Uri("Images\\3_u10484.png", UriKind.Relative));this.I ...

  2. 用C#读取图片的EXIF信息的方法

    using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Dr ...

  3. Android中读取图片EXIF元数据之metadata-extractor的使用

    一.引言及介绍 近期在开发中用到了metadata-extractor-xxx.jar 和 xmpcore-xxx.jar这个玩意, 索性查阅大量文章了解学习,来分享分享. 本身工作也是常常和处理大图 ...

  4. WPF 后台C#设置控件背景图片

    原文:WPF 后台C#设置控件背景图片 以前的程序中有做过,当时只是记得uri很长一大段就没怎么记.今天有人问了也就写下来.   这是一个Button,设置了Background后的效果. 前台的设置 ...

  5. Java读取图片exif信息实现图片方向自动纠正

    起因 一个对试卷进行OCR识别需求,需要实现一个功能,一个章节下的题目图片需要上下拼接合成一张大图,起初写了一个工具实现图片的合并,程序一直很稳定的运行着,有一反馈合成的图片方向不对,起初怀疑是本身图 ...

  6. 用WPF窗体打造个性化界面的图片浏览器

    原文:用WPF窗体打造个性化界面的图片浏览器 本文使用WPF窗体(XAML及C#)与Win Form控件(FolderBrowserDialog)结合的方式, 演示制作了一个简易漂亮的WPF图片浏览器 ...

  7. python 修改、读取图片元数据

    图片元数据 图片元数据(metadata)是嵌入到图片文件中的一些标签.比较像文件属性,但是种类繁多.常见的几种标准有: EXIF:通常被数码相机在拍摄照片时自动添加,比如相机型号.镜头.曝光.图片尺 ...

  8. HTML中上传与读取图片或文件(input file)----在路上(25)

    input file相关知识简例 在此介绍的input file相关知识为: 上传照片及文件,其中包括单次上传.批量上传.删除照片.增加照片.读取图片.对上传的图片或文件的判断,比如限制图片的张数.限 ...

  9. python 读取图片的尺寸、分辨率

    #需要安装PIL模块 #encoding=gbk#--------------------------------------------------------------------------- ...

  10. OpenGL——OpenCV与SOIL读取图片进行纹理贴图

    使用OpenCV读取图片代码如下 img = imread(m_fileName); if (img.empty()) { fprintf(stderr, "Can not load ima ...

随机推荐

  1. FFmpeg开发笔记(三)FFmpeg的可执行程序介绍

    ​外界对于FFmpeg主要有两种使用途径,一种是在命令行运行FFmpeg的可执行程序,该方式适合没什么特殊要求的普通场景:另一种是通过代码调用FFmpeg的动态链接库,由于开发者可以在C代码中编排个性 ...

  2. [Java]细节与使用经验

    [版权声明]未经博主同意,谢绝转载!(请尊重原创,博主保留追究权) https://www.cnblogs.com/cnb-yuchen/p/18032072 出自[进步*于辰的博客] 纯文字阐述,内 ...

  3. 在 M1 下搭建 DolphinScheduler 开发调试环境

    Apache DolphinScheduler 是一个分布式去中心化,易扩展的可视化 DAG 工作流任务调度系统.致力于解决数据处理流程中错综复杂的依赖关系,使调度系统在数据处理流程中开箱即用 M1 ...

  4. Redis高可用之战:主从架构

    ★ Redis24篇集合 1 主从模式介绍 在笔者的另外两篇文章 <Redis系列:RDB内存快照提供持久化能力>.<Redis稳定性之战:AOF日志支撑数据持久化>中,我们介 ...

  5. KingbaseES 实现 MySQL 函数 last_insert_id

    用户从mysql迁移到金仓数据库过程中,应用中使用了mysql函数last_insert_id()来获取最近insert的那行记录的自增字段值. mysql文档中关于函数的说明和例子: LAST_IN ...

  6. html中怎样获取子元素的索引位置

    jQuery 的 index() 方法返回指定元素相对于其他指定元素的索引值, 注意:索引值是从0开始计数的. 获得当前元素的索引值可用click事件触发 1 $(selector).click(fu ...

  7. #贪心#洛谷 6927 [ICPC2016 WF]Swap Space

    题目 分析 可以发现能将硬盘容量变大的优先,这种硬盘就是以格式化前的大小升序排序. 然后如果硬盘容量变小,那就是先填格式化后较大的硬盘(因为装完可以提供较大的空间) 代码 #include <c ...

  8. #线段树#LOJ 6029「雅礼集训 2017 Day1」市场

    题目 在长度为\(n(n\leq 10^5)\)的数列中, 需要满足区间加,区间下取整的操作 以及能够查询区间和以及区间最小值 除数\(d\)满足\(2\leq d\leq 10^9\) 加数\(c\ ...

  9. 本周四晚19:00知识赋能第八期第2课丨ArkUI自定义组件

     9月21日19:00~20:00,第八期知识赋能第2节直播就要开始啦!本次直播将为同学们带来涂鸦小游戏的趣味体验,让大家全面了解ArkUI框架的应用,帮助你们在自己已有专业的基础上拓宽知识边界,学习 ...

  10. 30分钟成为Contributor|如何多方位参与OpenHarmony开源贡献?

    如何优雅地参与开源贡献,向顶级开源项目提交 PR(Pull Request).战"码"先锋直播间第八期围绕"OpenAtom OpenHarmony(以下简称" ...