3

我在一个旧问题上使用以下答案中的代码来设置上传图像的缩略图。

https://stackoverflow.com/a/2001462/1593395

这完美地工作并填充图像,保持纵横比等,但如果我从手机上传图像,则通过此方法保存的缩略图图像将逆时针旋转 90 度。

你知道是什么原因造成的吗?原始图像只是使用 AjaxFileUpload1.SaveAs(MapPath("~/catalog/images/" & imageFilename)) (来自 AJAX Control 工具包)保存并以正确的方向显示。

谢谢

4

1 回答 1

7

可能是由于图像物理存储在其显示的不同方向上,例如,用相机侧面拍摄的 640*480 镜头可能会存储为 480*640,带有方向 exif 数据标志。

这很棒,因为 explorer/paint/photoshop/几乎每个查看者都会看到 exif 标志并在渲染之前对其进行旋转。但是,.net 图像类不(当您知道发生了什么时这似乎是合理的),因此您必须在新的缩略图图像上设置 exif 旋转属性(我不喜欢这样做,只是因为我不喜欢缩略图上有任何属性)或自己检查和旋转缩略图。

下面是一个粗略的方法来做到这一点。请注意,我在 c# 中提供了代码作为您引用的答案的修改版本,因为那也是 c#。转换到 vb.net 应该非常简单:)

  if (sourceImage.PropertyIdList.Contains(0x112)) //0x112 = Orientation
  {
     var prop = sourceImage.GetPropertyItem(0x112);
     if (prop.Type == 3 && prop.Len == 2)
     {
        UInt16 orientationExif = BitConverter.ToUInt16(sourceImage.GetPropertyItem(0x112).Value, 0);
        if (orientationExif == 8)
        {
           newImage.RotateFlip(RotateFlipType.Rotate270FlipNone);
        }
        else if (orientationExif == 3)
        {
           newImage.RotateFlip(RotateFlipType.Rotate180FlipNone);
        }
        else if (orientationExif == 6)
        {
           newImage.RotateFlip(RotateFlipType.Rotate90FlipNone);
        }
     }
  }

所以更新的 FixedSize 代码将是这样的:

static Image FixedSize(Image imgPhoto, int Width, int Height)
{
    int sourceWidth = imgPhoto.Width;
    int sourceHeight = imgPhoto.Height;
    int sourceX = 0;
    int sourceY = 0;
    int destX = 0;
    int destY = 0;

    float nPercent = 0;
    float nPercentW = 0;
    float nPercentH = 0;

    nPercentW = ((float)Width / (float)sourceWidth);
    nPercentH = ((float)Height / (float)sourceHeight);
    if (nPercentH < nPercentW)
    {
        nPercent = nPercentH;
        destX = System.Convert.ToInt16((Width -
                      (sourceWidth * nPercent)) / 2);
    }
    else
    {
        nPercent = nPercentW;
        destY = System.Convert.ToInt16((Height -
                      (sourceHeight * nPercent)) / 2);
    }

    int destWidth = (int)(sourceWidth * nPercent);
    int destHeight = (int)(sourceHeight * nPercent);

    Bitmap bmPhoto = new Bitmap(Width, Height,
                      PixelFormat.Format24bppRgb);
    bmPhoto.SetResolution(imgPhoto.HorizontalResolution,
                     imgPhoto.VerticalResolution);

    Graphics grPhoto = Graphics.FromImage(bmPhoto);
    grPhoto.Clear(Color.Red);
    grPhoto.InterpolationMode =
            InterpolationMode.HighQualityBicubic;

    grPhoto.DrawImage(imgPhoto,
        new Rectangle(destX, destY, destWidth, destHeight),
        new Rectangle(sourceX, sourceY, sourceWidth, sourceHeight),
        GraphicsUnit.Pixel);

    grPhoto.Dispose();

    //Rotate image to what is expected.
    if (imgPhoto.PropertyIdList.Contains(0x112)) //0x112 = Orientation
    {
       var prop = imgPhoto.GetPropertyItem(0x112);
       if (prop.Type == 3 && prop.Len == 2)
       {
          UInt16 orientationExif = BitConverter.ToUInt16(sourceImage.GetPropertyItem(0x112).Value, 0);
          if (orientationExif == 8)
          {
             bmPhoto.RotateFlip(RotateFlipType.Rotate270FlipNone);
          }
          else if (orientationExif == 3)
          {
             bmPhoto.RotateFlip(RotateFlipType.Rotate180FlipNone);
          }
          else if (orientationExif == 6)
          {
             bmPhoto.RotateFlip(RotateFlipType.Rotate90FlipNone);
          }
       }
    }

    return bmPhoto;
}

请注意,这并不涵盖所有 exif 方向,仅涵盖常见方向。

参考:

http://www.impulseadventure.com/photo/exif-orientation.html

http://msdn.microsoft.com/en-us/library/xddt0dz7.aspx

ps:这是我的第一个堆栈溢出答案,所以请在反馈中放轻松;)

于 2013-11-05T02:07:31.747 回答