.NET - Граница вокруг измененного размера изображения - PullRequest
5 голосов
/ 28 марта 2011

Я пытаюсь изменить размер изображения в .NET, но получаю слабую черную рамку вокруг измененного изображения. Я нашел пост - http://social.msdn.microsoft.com/forums/en-US/csharpgeneral/thread/cf765094-c8c1-4991-a1f3-cecdbd07ee15/, который от кого-то, кто сказал, что создание прямоугольника назначения больше, чем холст, работал, но это не работает для меня. Он освобождается от верхней и левой границ, но правая и нижняя часть все еще там, и имеют полный 1px толстый черный цвет.

Я что-то упустил? Мой код ниже.

Image image = ... // this is a valid image loaded from the source
Rectangle srcRectangle = new Rectangle(0,0,width, height);
        Size croppedFullSize = new Size(width+3,height+3);
        Rectangle destRect = new Rectangle(new Point(-1,-1), croppedFullSize);
        using(Bitmap newImage = new Bitmap(croppedFullSize.Width, croppedFullSize.Height, format))
        using(Graphics Canvas = Graphics.FromImage(newImage)) {
            Canvas.SmoothingMode = SmoothingMode.AntiAlias;
            Canvas.InterpolationMode = InterpolationMode.HighQualityBicubic;
            Canvas.PixelOffsetMode = PixelOffsetMode.HighQuality;
            Canvas.FillRectangle(Brushes.Transparent, destRect);
            Canvas.DrawImage(image, destRect, srcRectangle, GraphicsUnit.Pixel);


            newImage.Save(filename, image.RawFormat);
        }

Ответы [ 3 ]

11 голосов
/ 07 августа 2012

Просто предоставьте методу DrawImage экземпляр ImageAttributes, для которого для WrapMode установлено значение TileFlipXY.Это предотвратит смешивание края с фоновым цветом.

Для примера кода, который не пропускает память, как другие ответы здесь, см. Эту суть

3 голосов
/ 28 марта 2011

Попробуйте так, я думаю, у меня никогда не было черной границы ...

Если вы хотите использовать библиотеки System.Drawing:

using (var sourceBmp = new Bitmap(sourcePath))
{
  decimal aspect = (decimal)sourceBmp.Width / (decimal)sourceBmp.Height;
  int newHeight = (int)(newWidth / aspect);

   using (var destinationBmp = new Bitmap(newWidth, newHeight))
   {
     using (var destinationGfx = Graphics.FromImage(destinationBmp))
     {
       destinationGfx.InterpolationMode = InterpolationMode.HighQualityBicubic;
       destinationGfx.DrawImage(sourceBmp, new Rectangle(0, 0, destinationBmp.Width, destinationBmp.Height));
       destinationBmp.Save(destinationPath, ImageFormat.Jpeg);
      }
    }
}

или вы можете сделатьто же самое с wpf, вот так:

using (var output = new FileStream(outputPath, FileMode.CreateNew, FileAccess.ReadWrite, FileShare.None))
{
   var imageDecoder = BitmapDecoder.Create(inputStream, BitmapCreateOptions.PreservePixelFormat, BitmapCacheOption.None);
   var imageFrame = imageDecoder.Frames[0];

   decimal aspect = (decimal)imageFrame.Width / (decimal)imageFrame.Height;
   var height = (int)(newWidth / aspect);

   var imageResized = new TransformedBitmap(imageFrame,new ScaleTransform(
                                                                 newWidth / imageFrame.Width * Dpi / imageFrame.DpiX,
                                                                 height / imageFrame.Height * Dpi / imageFrame.DpiY, 0, 0));

   var targetFrame = BitmapFrame.Create(imageResized);

   var targetEncoder = new JpegBitmapEncoder();
   targetEncoder.Frames.Add(targetFrame);
   targetEncoder.QualityLevel = 80;
   targetEncoder.Save(output);
}

Я рекомендую способ WPF.Сжатие и качество кажутся лучше ...

1 голос
/ 13 февраля 2017

Для меня это был плохой параметр Bitmap.Вместо этого:

new Bitmap(width, height, PixelFormat.Format32bppPArgb);

Просто удалите PixelFormat к этому:

new Bitmap(width, height);

И тогда все было в порядке.верхняя и левая границы.Затем я попытался g.PixelOffsetMode = PixelOffsetMode.HighQuality;который казался прекрасным вначале.Но потом я заметил светло-серые границы вокруг всего изображения.

...