Introduction
It's not often that we work with extremely low-resolution, or pixelated images, but when we do we may have differing expectations of what that image should look like.
In a recent project I wanted to show the pixelation of the images, but when shown in the default PictureBox
, the pixelation could not be seen, because the control was smoothing the pixelation.
This is what I wanted (InterpolationType.NearestNeighbor
):
data:image/s3,"s3://crabby-images/c34ff/c34ffc90321f1629993a2c8ede6347c79ac91eab" alt=""
This is what I got (InterpolationType.Default
):
data:image/s3,"s3://crabby-images/fb9f6/fb9f6e32e1873a269a49df0676a68ff4b1eb5540" alt=""
Background
Windows.Forms.Controls.PictureBox
is the standard control that is used to display images within a Windows Forms application. It does a decent job of this and offers some customization options, such as SizeMode
, to set how an image should fit in the box.
When displaying an image at any size other than the images original size i.e. pixel-to-pixel, the extra pixels used to turn a low-resolution image into a higher one, and vise-versa have to come from somewhere. This is called interpolation, and it turns out there are many algorithms to accomplish this.
PixelBox
is a custom control that is derived from PictureBox
. It adds the InterpolationMode
property, which has the effect of setting the GDI+ interpolation mode before the image is drawn onto the control's surface. The property is a value from the InterpolationMode enumeration.
The code
The changes to PictureBox
are quite simple. The entire class is shown below:
using System.ComponentModel;
using System.Drawing.Drawing2D;
using System.Windows.Forms;
namespace RedCell.UI.Controls
{
public class PixelBox : PictureBox
{
#region Initialization
public PixelBox ()
{
InterpolationMode = InterpolationMode.Default;
}
#endregion
#region Properties
[Category("Behavior")]
[DefaultValue(InterpolationMode.Default)]
public InterpolationMode InterpolationMode { get; set; }
#endregion
#region Overrides of PictureBox
protected override void OnPaint (PaintEventArgs pe)
{
pe.Graphics.InterpolationMode = InterpolationMode;
base.OnPaint(pe);
}
#endregion
}
}
</see>
Demonstration
The demonstration application demonstrates any of the interpolation modes on a pixelated image. In addition to those shown above, here are the other modes for comparison:
InterpolationMode.Low
data:image/s3,"s3://crabby-images/e42e7/e42e77122aaef2b39ee69004cefb9ed0ba314a41" alt=""
InterpolationMode.High
data:image/s3,"s3://crabby-images/0fb2d/0fb2da4bcaa1c7e94d1f4b8c201050915cc6e76f" alt=""
InterpolationMode.Bilinear
data:image/s3,"s3://crabby-images/b1316/b131623e93603adf12f7ee1302a5f9084fbbb980" alt=""
InterpolationMode.Bicubic
data:image/s3,"s3://crabby-images/ee811/ee811ebc780fd1316cb14db59c419f0999e9cd3e" alt=""
InterpolationMode.HighQualityBilinear
data:image/s3,"s3://crabby-images/f448f/f448f5ce69e08aa6327d67bd31d699c85111d619" alt=""
InterpolationMode.HighQualityBicubic
data:image/s3,"s3://crabby-images/1b89f/1b89f0cb0c7e7a5acb9d29d7e0b3d1df55f749ea" alt=""
Conclusion
By subclassing PictureBox
and overriding its OnPaint
method we can set the GDI+ InterpolationMode
property, giving us greater control over the algorithm, quality, and speed of rendering.
History
- January 28, 2014
- Initial publication.