使用C#进行图片转换格式,缩放,自动旋转,保留exif(转载)
这几天心血来潮做了一个批量图片缩放,转换格式,并且可以根据exif的信息旋转图片,校正exif信息后保存的小程序.根据配置文件 指定需要的功能.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
|
using System; using System.IO; using System.Drawing.Imaging; using System.Drawing; using System.Drawing.Drawing2D; using System.Runtime.InteropServices; using System.Text; namespace PhotoUtil { class Program { [DllImport( "kernel32" )] private static extern long GetPrivateProfileString( string section, string key, string def, StringBuilder retVal, int size, string filePath); private const String configFile = "Config.ini" ; public static void Main( string [] args) { DirectoryInfo workingDir = new DirectoryInfo(ReadConfig( "General" , "WorkingDir" ,Environment.CurrentDirectory)); if (!workingDir.Exists) { workingDir = new DirectoryInfo(Environment.CurrentDirectory); } int quality= int .Parse(ReadConfig( "General" , "Quality" , "85" )); bool needResize = Boolean.Parse(ReadConfig( "ResizeImage" , "Enable" , "false" )); int newWidth = int .Parse(ReadConfig( "ResizeImage" , "NewWidth" , "800" )); int newHeight = int .Parse(ReadConfig( "ResizeImage" , "NewHeight" , "600" )); bool padding = Boolean.Parse(ReadConfig( "ResizeImage" , "Padding" , "false" )); bool needRotate = Boolean.Parse(ReadConfig( "RotateImage" , "Enable" , "true" )); FileInfo[] files = workingDir.GetFiles(); DirectoryInfo output = workingDir.CreateSubdirectory(DateTime.Now.ToString( "yyyyMMdd" ) + "转换\r foreach (FileInfo i in files) { String type = i.Extension.ToLower(); if (type.Contains( "jpg" ) || type.Contains( "jpeg" ) || (type.Contains( "png" )) || type.Contains( "tif" ) || type.Contains( "bmp" )) { Image img = Image.FromFile(i.FullName); if (needResize) { Console.WriteLine( "Resizing " + i.FullName); ResizeImage( ref img, newWidth, newHeight, padding); } if (needRotate) { Console.WriteLine( "Rotating " + i.FullName); RotateImage(img); } SaveAs(img, output.FullName+ "\\\\" +i.Name, quality); } } Console.ReadLine(); } private static void SaveAs(Image img, string dest, long quality) { if (quality > 100 || quality < 1) { quality = 85; } EncoderParameters para = new EncoderParameters(); para.Param[0] = new EncoderParameter(System.Drawing.Imaging.Encoder.Quality, quality); String extension = new FileInfo(dest).Extension; ImageCodecInfo info = GetImageCodecInfoByExtension(extension); if (info != null ) { img.Save(dest, info, para); } else { throw new Exception( "Unrecognized format \\" " + extension + " \\"!\r } } private static void ResizeImage( ref Image image, int expectDestWidth, int expectDestHeight, bool padding) { PropertyItem[] exif = image.PropertyItems; int targetWidth = 0; int targetHeight = 0; double srcHWRate = ( double )image.Width / ( double )image.Height; double expectHWRate = ( double )expectDestWidth / ( double )expectDestHeight; if (srcHWRate > expectHWRate) { targetWidth = expectDestWidth; targetHeight = System.Convert.ToInt32(Math.Round(expectDestWidth / srcHWRate, 0)); } else { targetHeight = expectDestHeight; targetWidth = System.Convert.ToInt32(Math.Round(expectDestHeight * srcHWRate, 0)); } Image bitmap = null ; if (!padding) { bitmap = new Bitmap(targetWidth, targetHeight); } else { bitmap = new Bitmap(expectDestWidth, expectDestHeight); } Graphics g = Graphics.FromImage(bitmap); g.CompositingQuality = CompositingQuality.HighQuality; g.InterpolationMode = InterpolationMode.HighQualityBicubic; g.SmoothingMode = SmoothingMode.HighQuality; g.DrawImage(image, new Rectangle(0, 0, bitmap.Width, bitmap.Height), new Rectangle(0, 0, image.Width, image.Height), GraphicsUnit.Pixel); foreach (PropertyItem i in exif) { if (i.Id == 40962) { i.Value = BitConverter.GetBytes(targetWidth); } else if (i.Id == 40963) { i.Value = BitConverter.GetBytes(targetHeight); } bitmap.SetPropertyItem(i); } g.Dispose(); image.Dispose(); image = bitmap; } private static string ReadConfig(String Section, String Key, String defaultValue) { if (File.Exists(configFile)) { StringBuilder temp = new StringBuilder(1024); GetPrivateProfileString(Section, Key, String.Empty, temp, 1024, new FileInfo(configFile).FullName); if (!String.IsNullOrEmpty(temp.ToString())) { return temp.ToString(); } else { return defaultValue; } } else { return defaultValue; } } public static void RotateImage(Image img) { PropertyItem[] exif = img.PropertyItems; byte orientation = 0; foreach (PropertyItem i in exif) { if (i.Id == 274) { orientation = i.Value[0]; i.Value[0] = 1; img.SetPropertyItem(i); } } switch (orientation) { case 2: img.RotateFlip(RotateFlipType.RotateNoneFlipX); break ; case 3: img.RotateFlip(RotateFlipType.Rotate180FlipNone); break ; case 4: img.RotateFlip(RotateFlipType.RotateNoneFlipY); break ; case 5: img.RotateFlip(RotateFlipType.Rotate90FlipX); break ; case 6: img.RotateFlip(RotateFlipType.Rotate90FlipNone); break ; case 7: img.RotateFlip(RotateFlipType.Rotate270FlipX); break ; case 8: img.RotateFlip(RotateFlipType.Rotate270FlipNone); break ; default : break ; } foreach (PropertyItem i in exif) { if (i.Id == 40962) { i.Value = BitConverter.GetBytes(img.Width); } else if (i.Id == 40963) { i.Value = BitConverter.GetBytes(img.Height); } } } private static ImageCodecInfo GetImageCodecInfoByExtension(String extension) { ImageCodecInfo[] list = ImageCodecInfo.GetImageEncoders(); foreach (ImageCodecInfo i in list) { if (i.FilenameExtension.ToLower().Contains(extension.ToLower())) { return i; } } return null ; } } } |
配置文件名称:Config.ini,放在和程序同目录下,格式如下:
[General]
#工作目录
WorkingDir=.
#输出质量,1-100范围内的整数
Quality=80
[ResizeImage]
#是否启用
Enable=true
#新的宽值和高值
NewWidth=1024
NewHeight=768
#是否对超出原来比例的部分进行填充
Padding=false
#是否使用Exif中的旋转信息对图片进行旋转
[RotateImage]
Enable=true
1.保留exif信息的技巧是获取原始的Image.PropertyItems,对新图片添加即可.
2.关于Exif Orientation标志的定义 http://sylvana.net/jpegcrop/exif_orientation.html
3.变更jpeg输出质量的方法:
EncoderParameters para = new EncoderParameters();
para.Param[0] = new EncoderParameter(System.Drawing.Imaging.Encoder.Quality, quality);
//这里的quality是从1-100的long值,一开始想当然的以为1-100的值就用了byte,结果出错了.
最后调用Image.Save(String, ImageCodecInfo, EncoderParameters)来进行输出.
4.进行缩放/旋转后需要调整原始的exif信息,id查文档可知.列出一些
id=40962,width
id=40963,height
id=274,orientation type
对PropertyItem信息设置完了别忘了Image.SetPropertyItem(PropertyItem)添加到image中,我出一个很 傻的bug,就是在修改了orientation type=1之后没有SetPropertyItem,于是导致图片实际被旋转至正确角度了,但是用ACDSee打开后可以看到exif的旋转信息依旧是 原来的值.
使用C#进行图片转换格式,缩放,自动旋转,保留exif(转载)的更多相关文章
- 23.Quick QML-简单且好看的图片浏览器-支持多个图片浏览、缩放、旋转、滑轮切换图片
之前我们已经学习了Image.Layout布局.MouseArea.Button.GroupBox.FileDialog等控件. 所以本章综合之前的每章的知识点,来做一个图片浏览器,使用的Qt版本为Q ...
- android 图片的平移,缩放和旋转
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools= ...
- ABBYY如何把图片转换成pdf格式
在制作工作文件的时候,有时候会遇到需要进行文件格式转换的情况,比较常见的文件格式转换就包含了Office与pdf格式之间的转换.但除此之外,图片与pdf格式也是可以进行转换的,那么图片要怎么操作,才能 ...
- Android代码中动态设置图片的大小(自动缩放),位置
项目中需要用到在代码中动态调整图片的位置和设置图片大小,能自动缩放图片,用ImageView控件,具体做法如下: 1.布局文件 <RelativeLayout xmlns:android=&qu ...
- jpg、png格式的图片转换成webp后颜色失真的问题
今天简单的试用了一下 cweb.exe 将 jpg, png 格式的图片转换成 webp 格式. 我今天下载的是当前最新版:1.0.0 cwebp 3.jpg -q 85 -o 3.webp 发现图 ...
- 如何将.jpg图片 转换成.eps 格式图片
在使用latex写作论文的时候,需要插入一些图片,但是往往有些图片不是eps格式的.虽然网上有如何插入jpg格式的图片方法,但是经过我实验后发现都不太管用.最后找到一个比较靠谱的方法,使用latx本身 ...
- centos下 将(jgp、png)图片转换成webp格式
由于项目要求需要将jpg.png类型的图片 转换成webp格式,最开始使用了php gd类库里 imagewebp 方法实现,结果发现转换成的webp格式文件会偶尔出现空白内容的情况.像创建了一个透 ...
- Android动画及图片的缩放和旋转
Android动画有2种,一种是Tween Animation,另一种是Frame Animation,先说说Tween动画吧. Tween动画是对视图对象中的内容进行一系列简单的转换,比如位置的移动 ...
- xnconvert 图片转换工具
xnconvert是一款简单高效的图片转换工具.xnconvert能够批量地进行图片格式转换,并具有一定的图片处理功能,可以增加水印.特效,支持放大缩小.旋转等. xnconvert功能介绍: 你可以 ...
随机推荐
- Dottrace跟踪代码执行时间
当自己程序遇到性能问题,比如请求反应缓慢,怎么分析是哪里出了问题呢?dottrace可以帮助.net程序跟踪出代码里每个方法的执行时间,这样让我们更清晰的看出是哪里执行时间过长,然后再分析应该怎样解决 ...
- java中的静态代码块、构造代码块、构造方法
运行下面这段代码,观察其结果: package com.test; public class HelloB extends HelloA { public HelloB() { } { System. ...
- python selenuim使用代理的方式
一.FireFox浏览器 myProxy = "60.195.250.55:80" proxy = Proxy({ 'proxyType': ProxyType.MANUAL, ' ...
- selection伪元素小解
上一篇:<RGBA与Opacity区别小解> p{font-size:14px;} 今天说一个简单的伪元素::selection,它的用武之地仅在于改变选中文本时文本的颜色和文本背景颜色. ...
- IE8/9的console之坑
这几天遇到个深坑,在改别人代码时,发现ajax在ie8下请求不成功.清理了缓存后,可以请求成功!(清理缓存只是表象而已,后文说原因) 后来慢慢看代码,发现ajax成功回调了!在success回调里,我 ...
- EntityFramework中的线程安全,又是Dictionary
继上次记一次w3wp占用CPU过高的解决过程(Dictionary和线程安全)后又再次与Dictionary博弈,这一次是在EntityFramework中的Dictionary. 从一个异常说起 这 ...
- EntityFramework_MVC4中EF5 新手入门教程之七 ---7.通过 Entity Framework 处理并发
在以前的两个教程你对关联数据进行了操作.本教程展示如何处理并发性.您将创建工作与各Department实体的 web 页和页,编辑和删除Department实体将处理并发错误.下面的插图显示索引和删除 ...
- WCF入门(22)
前言 本还想写一集WCF入门教程的,心情实在不好,明天又还有面试,改天再写吧. 说一下今天遇到的入职坑.面试能坑,上班能坑,完全没想到入职也能坑.切身经历. 今年10月份想换工作,更新了一下简历,接到 ...
- oracle-7参数文件的管理
参数文件的管理:1.参数文件的作用:记录数据库的配置的 (1)pfile ---> 文本文件 (2)spfile --->服务器的参数文件(二进制的) 两个参数文件的区别: pfile ...
- iOS开发小技巧--父子控制器练习中get到的技能,控制核心动画的范围
一.未经过处理的动画是这样的,自定义的导航按钮也一起跟着转起来了. 二.自己想要的效果 三.实现这种效果的思想:核心动画要添加到view的layer上面,刚开始的情况是讲核心动画添加到了整个大view ...