随笔 - 2146  文章 - 97 评论 - 11760 trackbacks - 253


通过 IGPImage.GetEncoderParameterList 可以获取指定编码格式的参数列表;

通过此列表可以遍历出各参数的指针: PGPNativeEncoderParameter(TGPNativeEncoderParameter 的指针);

TGPNativeEncoderParameter 是一个结构体:
TGPNativeEncoderParameter = record
  Guid: TGUID;                              { 参数标识 }
  NumberOfValues: ULONG;                    { 参数数组的元素数 }
  ValueType: TGPEncoderParameterValueType;  { 参数类型 }
  Value: Pointer;                           { 参数数据指针 }
end;

//其中的 TGPEncoderParameterValueType 是个枚举, 枚举值有:
EncoderParameterValueTypeByte          = 1 { 字节数组 }
EncoderParameterValueTypeASCII         = 2 { PAnsiChar }
EncoderParameterValueTypeShort         = 3 { Word }
EncoderParameterValueTypeLong          = 4 { Cardinal }
EncoderParameterValueTypeRational      = 5 { Cardinal/Cardinal; 第一个数是分子, 第二个数是分母 }
EncoderParameterValueTypeLongRange     = 6 { 一对 Cardinal, 表示一个数值范围 }
EncoderParameterValueTypeUndefined     = 7 { 可包含任何数据类型的字节数组 }
EncoderParameterValueTypeRationalRange = 8 { 四个整数: Cardinal/Cardinal, Cardinal/Cardinal }
EncoderParameterValueTypePointer       = 9 { 指针 }

//EncoderParameterValueTypeRationalRange 中的四个整数通过分数运算得到的两个值: 最小值...最大值.


每个编码器的参数肯定会有区别, 下面代码获取了 JPEG 编码器所能支持的参数信息:
uses GdiPlus;

procedure TForm1.Button1Click(Sender: TObject);
var
  Image: IGPImage;
  Parameters: IGPEncoderParameters;
  Param: PGPNativeEncoderParameter;
begin
  Image := TGPBitmap.Create(1, 1);
  Parameters := Image.GetEncoderParameterList(TGPImageFormat.Jpeg.CodecId);

  Memo1.Clear;
  for Param in Parameters do with Memo1.Lines do
  begin
    Add(Format('Guid: %s', [GUIDToString(Param.Guid)]));
    Add(Format('NumberOfValues: %d', [Param.NumberOfValues]));
    Add(Format('ValueType: %d', [Ord(Param.ValueType)]));
    Add(Format('Value: $%p', [Param.Value]));
    Add(EmptyStr);
  end;
end;
(* 结果:
  Guid: {8D0EB2D1-A58E-4EA8-AA14-108074B7B6F9}
  NumberOfValues: 5
  ValueType: 4
  Value: $00AD8190

  Guid: {1D5BE4B5-FA4A-452D-9CDD-5DB35105E7EB}
  NumberOfValues: 1
  ValueType: 6
  Value: $00AD81A4

  Guid: {EDB33BCE-0266-4A77-B904-27216099E717}
  NumberOfValues: 0
  ValueType: 3
  Value: $00AD81AC

  Guid: {F2E455DC-09B3-4316-8260-676ADA32481C}
  NumberOfValues: 0
  ValueType: 3
  Value: $00AD81AC
*)


IGPImage.GetEncoderParameterList 方法得到的类型是: IGPEncoderParameters;

IGPImage.Save 方法就有一个 IGPEncoderParameters 类型的默认参数, 通过它可以传入编码参数.

编码参数有很多类型, 譬如 EncoderQuality 是决定图片压缩比率的.

下面的例子在保存 JPG 文件时使用了三种不同的质量参数(压缩级别):



uses GdiPlus;

procedure TForm1.Button1Click(Sender: TObject);
var
  Prams: IGPEncoderParameters;
  Image: IGPImage;
  Graphics: IGPGraphics;
  Quality: Integer;
begin
  ChDir('C:\GdiPlusImg\');
  Image := TGPImage.Create('GrapeBunch.bmp');

  Prams := TGPEncoderParameters.Create;
  Quality := 1;
  Prams.Add(EncoderQuality, Quality);
  Image.Save('GrapeBunch_1.jpg', TGPImageFormat.Jpeg, Prams);

  Prams.Clear;
  Quality := 50;
  Prams.Add(EncoderQuality, Quality);
  Image.Save('GrapeBunch_50.jpg', TGPImageFormat.Jpeg, Prams);

  Prams.Clear;
  Quality := 100;
  Prams.Add(EncoderQuality, Quality);
  Image.Save('GrapeBunch_100.jpg', TGPImageFormat.Jpeg, Prams);

  //显示
  Graphics := TGPGraphics.Create(Handle);
  Image := TGPImage.Create('GrapeBunch_1.jpg');
  Graphics.DrawImage(Image, 10, 10);

  Graphics.TranslateTransform(Image.Width + 10, 0);
  Image := TGPImage.Create('GrapeBunch_50.jpg');
  Graphics.DrawImage(Image, 10, 10);

  Graphics.TranslateTransform(Image.Width + 10, 0);
  Image := TGPImage.Create('GrapeBunch_100.jpg');
  Graphics.DrawImage(Image, 10, 10);
end;


IGPEncoderParameters 的成员:
IGPEncoderParameters.GetEnumerator;
IGPEncoderParameters.Clear;
IGPEncoderParameters.Add();
IGPEncoderParameters.Count;
IGPEncoderParameters.Param[];
IGPEncoderParameters.NativeParams;

//其中的 Add 方法有多种重载, 这便于添加各种类型的数据; 参数类型常数:
EncoderCompression      { 压缩 }
EncoderColorDepth       { 颜色深度 }
EncoderScanMethod       { 扫描方法 }
EncoderVersion          { 版本 }
EncoderRenderMethod     { 呈现方法 }
EncoderQuality          { 质量 }
EncoderTransformation   { 转换 }
EncoderLuminanceTable   { 亮度表 }
EncoderChrominanceTable { 色度表 }
EncoderSaveFlag         { 保存标志 }
CodecIImageBytes        {  }
{ 下面是 GDI+1.1 才开始支持的: }
EncoderColorSpace       {  }
EncoderImageItems       {  }
EncoderSaveAsCMYK       {  }


五种编码器(BMP、JPEG、GIF、TIFF、PNG)分别能支持哪些 "参数类型" 呢?
支持的参数类型的参数又是什么格式的呢? 尽管 Add 方法已准备好了多种重载, 用哪个呢?

下面的程序列出了各种编码器的参数信息:

uses GdiPlus;

const
  ParamValueTypeArr: array[1..9] of string = (
    'ValueTypeByte',
    'ValueTypeASCII',
    'ValueTypeShort',
    'ValueTypeLong',
    'ValueTypeRational',
    'ValueTypeLongRange',
    'ValueTypeUndefined',
    'ValueTypeRationalRange',
    'ValueTypePointer'
  );

//自定义函数
function GetGuidName(g: TGUID): string;
var
  s: string;
begin
  s := EmptyStr;
  if IsEqualGUID(g, EncoderCompression) then s := 'EncoderCompression';
  if IsEqualGUID(g, EncoderColorDepth) then s := 'EncoderColorDepth';
  if IsEqualGUID(g, EncoderScanMethod) then s := 'EncoderScanMethod';
  if IsEqualGUID(g, EncoderVersion) then s := 'EncoderVersion';
  if IsEqualGUID(g, EncoderRenderMethod) then s := 'EncoderRenderMethod';
  if IsEqualGUID(g, EncoderQuality) then s := 'EncoderQuality';
  if IsEqualGUID(g, EncoderTransformation) then s := 'EncoderTransformation';
  if IsEqualGUID(g, EncoderLuminanceTable) then s := 'EncoderLuminanceTable';
  if IsEqualGUID(g, EncoderChrominanceTable) then s := 'EncoderChrominanceTable';
  if IsEqualGUID(g, EncoderSaveFlag) then s := 'EncoderSaveFlag';
  if IsEqualGUID(g, CodecIImageBytes) then s := 'CodecIImageBytes';
  {$IF (GDIPVER >= $0110)}
  if IsEqualGUID(g, EncoderColorSpace) then s := 'EncoderColorSpace';
  if IsEqualGUID(g, EncoderImageItems) then s := 'EncoderImageItems';
  if IsEqualGUID(g, EncoderSaveAsCMYK) then s := 'EncoderSaveAsCMYK';
  {$IFEND}
  Result := s;
end;

procedure TForm1.Button1Click(Sender: TObject);
var
  Image: IGPImage;
  Parameters: IGPEncoderParameters;
  Param: PGPNativeEncoderParameter;
  Encoder: IGPImageCodecInfo;
begin
  Image := TGPBitmap.Create(1, 1);

  Memo1.Clear;
  for Encoder in TGPImageCodecInfo.GetImageEncoders do with Memo1.Lines do
  begin
    Parameters := Image.GetEncoderParameterList(Encoder.ClsId);

    Add('----------------------');
    Add(Format('编码器类型: %s', [Encoder.FormatDescription]));
    if Parameters.Count = 0 then
    begin
      Add('无参数');
      Add(EmptyStr);
      Continue;
    end;

    Add(Format('参数个数: %d', [Parameters.Count]));
    for Param in Parameters do
    begin
      Add(Format('参数类型: %s', [GetGuidName(Param.Guid)]));
      Add(Format('参数值类型: %s', [ParamValueTypeArr[Ord(Param.ValueType)]]));
      Add(Format('参数值个数: %d', [Param.NumberOfValues]));
      Add(Format('参数值指针: $%p', [Param.Value]));
      Add(EmptyStr);
    end;
  end;
end;
(* 显示结果:
  ----------------------
  编码器类型: BMP
  无参数

  ----------------------
  编码器类型: JPEG
  参数个数: 4
  参数类型: EncoderTransformation
  参数值类型: ValueTypeLong
  参数值个数: 5
  参数值指针: $00AD8190

  参数类型: EncoderQuality
  参数值类型: ValueTypeLongRange
  参数值个数: 1
  参数值指针: $00AD81A4

  参数类型: EncoderLuminanceTable
  参数值类型: ValueTypeShort
  参数值个数: 0
  参数值指针: $00AD81AC

  参数类型: EncoderChrominanceTable
  参数值类型: ValueTypeShort
  参数值个数: 0
  参数值指针: $00AD81AC

  ----------------------
  编码器类型: GIF
  无参数

  ----------------------
  编码器类型: TIFF
  参数个数: 3
  参数类型: EncoderCompression
  参数值类型: ValueTypeLong
  参数值个数: 5
  参数值指针: $00AD8190

  参数类型: EncoderColorDepth
  参数值类型: ValueTypeLong
  参数值个数: 5
  参数值指针: $00AD81A4

  参数类型: EncoderSaveFlag
  参数值类型: ValueTypeLong
  参数值个数: 1
  参数值指针: $00AD81B8

  ----------------------
  编码器类型: PNG
  无参数
*)


从上面例子可以看出:

BMP、GIF、PNG 三种编码器没有编码参数(GIF 在 GDI+1.1 中是不是支持还没有测试).

JPEG 支持:
EncoderTransformation (转换)
EncoderQuality (质量)
EncoderLuminanceTable (亮度表)
EncoderChrominanceTable (色度表)

TIFF 支持:
EncoderCompression (压缩)
EncoderColorDepth (颜色深度)
EncoderSaveFlag (保存标志)

进而可以得知:

JPEG 编码的参数类型 EncoderTransformation 的可选值是:
TGPEncoderValue(13): EncoderValueTransformRotate90
TGPEncoderValue(14): EncoderValueTransformRotate180
TGPEncoderValue(15): EncoderValueTransformRotate270
TGPEncoderValue(16): EncoderValueTransformFlipHorizontal
TGPEncoderValue(17): EncoderValueTransformFlipVertical

JPEG 编码的参数类型 EncoderQuality 的可选值是: 0..100

TIFF 编码的参数类型 EncoderCompression 的可选值是:
TGPEncoderValue(2): EncoderValueCompressionLZW
TGPEncoderValue(3): EncoderValueCompressionCCITT3
TGPEncoderValue(4): EncoderValueCompressionCCITT4
TGPEncoderValue(5): EncoderValueCompressionRle
TGPEncoderValue(6): EncoderValueCompressionNone

TIFF 编码的参数类型 EncoderColorDepth 的可选值是: 1,4,8,24,32

这些值可以从下面程序获取:

uses GdiPlus;

var
  Image: IGPImage;
  Parameters: IGPEncoderParameters;
  p: PCardinal;

procedure TForm1.FormCreate(Sender: TObject);
begin
  Image := TGPBitmap.Create(1, 1);
end;

{$PointerMath On}
procedure TForm1.Button1Click(Sender: TObject);
begin
  Parameters := Image.GetEncoderParameterList(TGPImageFormat.Jpeg.CodecId);
  p := Parameters[0].Value; { EncoderTransformation }
  ShowMessageFmt('%d,%d,%d,%d,%d', [p[0],p[1],p[2],p[3],p[4]]); { 13,14,15,16,17 }
end;

procedure TForm1.Button2Click(Sender: TObject);
begin
  Parameters := Image.GetEncoderParameterList(TGPImageFormat.Jpeg.CodecId);
  p := Parameters[1].Value; { EncoderQuality }
  ShowMessageFmt('%d..%d', [p[0],p[1]]); { 0..100 }
end;

procedure TForm1.Button3Click(Sender: TObject);
begin
  Parameters := Image.GetEncoderParameterList(TGPImageFormat.Tiff.CodecId);
  p := Parameters[0].Value; { EncoderCompression }
  ShowMessageFmt('%d,%d,%d,%d,%d', [p[0],p[1],p[2],p[3],p[4]]); { 2,3,5,4,6 }
end;

procedure TForm1.Button4Click(Sender: TObject);
begin
  Parameters := Image.GetEncoderParameterList(TGPImageFormat.Tiff.CodecId);
  p := Parameters[1].Value; { EncoderColorDepth }
  ShowMessageFmt('%d,%d,%d,%d,%d', [p[0],p[1],p[2],p[3],p[4]]); { 1,4,8,24,32 }
end;


下面例子通过设置 JPEG 的 EncoderTransformation 编码参数, 保存了旋转后的图片:



uses GdiPlus;

procedure TForm1.Button1Click(Sender: TObject);
var
  Graphics: IGPGraphics;
  Prams: IGPEncoderParameters;
  Image: IGPImage;
begin
  ChDir('C:\GdiPlusImg\');
  Image := TGPImage.Create('Grapes.jpg');

  Prams := TGPEncoderParameters.Create;
  Prams.Add(EncoderTransformation, EncoderValueTransformRotate90);
  Image.Save('Grapes_Rotate90.jpg', TGPImageFormat.Jpeg, Prams);

  Graphics := TGPGraphics.Create(Handle);
  Graphics.DrawImage(Image, 10, 10, Image.Width, Image.Height);
end;

//为什么只有 JPEG 提供这种编码参数呢? 因为 jpg 文件在自动保存过程中会降低品质, 通过这种变换则不会.


另外:

1、编码参数值类型很多是 Cardinal, 这都是官方资料上的; 但 Add 函数中要的是 Integer 类型.

2、从上面获取的信息可以知道, GDI+1.0 还无法写入 gif 动画; GDI+1.1 能不能还没有测试.

3、通过设置 TIFF 编码器的 EncoderSaveFlag 类型参数可保存多页的 TIFF 文件, 下次接上.
posted on 2009-12-29 17:25 万一 阅读(...) 评论(...) 编辑 收藏