### Article Purpose

This article serves to provides a detailed discussion and implementation of a ** Stained Glass Image Filter**. Primary topics explored include: Creating Voronoi Diagrams, Pixel Coordinate distance calculations implementing Euclidean, Manhattan and Chebyshev methods. In addition, this article explores Gradient Based Edge Detection implementing thresholds.

*Zurich: Block Size 15, Factor 4, Euclidean*

### Sample Source Code

This article is accompanied by a sample source code Visual Studio project which is available for download here.

### Using the Sample Application

This article’s accompanying sample source code includes a Windows Forms based sample application. The sample application provides an implementation of the concepts explored by this article. Concepts discussed can be easily replicated and tested by using the sample application.

Source/input image files can be specified from the local file system when clicking the ** Load Image** button. Additionally users also have the option to save resulting filtered images by clicking the

**button.**

*Save Image*The sample application through its user interface allows a user to specify several filter configuration options. Two main categories of configuration options have been defined as ** Block Properties** and

**.**

*Edge Properties*** Block Properties** relate to the process of rendering Voronoi Diagrams. The following configuration options have been implemented:

During the process of rendering a Voronoi Diagram regions or blocks of equal shape and size have to be defined. These uniform regions/blocks form the basis of rendering uniquely shaped regions later on. The*Block Size –*option determines the width and height of an individual region/block. Larger values result in larger non-uniform regions being rendered. Smaller values in return result in smaller non-uniform regions being rendered.*Block Size*The*Distance Factor –*option determines the extent to which a pixel’s containing region will be calculated. Possible values range from 1 to 4 inclusive. A*Distance Factor*value of 4 equates to precise calculation of a pixel’s containing region, whereas a value of 1 results in containing regions often registering pixels that should be part of a neighbouring region. Values closer to 4 result in more varied region shapes. Values closer to 1 result in regions being rendered having more of a uniform shape/pattern.*Distance Factor*The distance between a pixel’s coordinates and a region’s outline determines whether that pixel should be considered part of a region. The sample application implements three different methods of calculating pixel distance: Euclidean, Manhattan and Chebyshev methods. Each result in region shapes being rendered differently.*Distance Formula –*

*Salzburg: Block Size 20, Factor 1, Chebyshev, Edge Threshold 2*

** Edge Properties** relate to the implementation of

**. Edge detection is an optional filter and can be enabled/disabled through the user interface, The implementation of edge detection serves to highlight/outline regions rendered as part of a Voronoi Diagram. The configuration options implemented are:**

*Image Gradient Based Edge Detection*Boolean value indicating whether or not edge detection should be applied*Highlight Edges –*In calculating image edges a threshold value determines if a pixel forms part of an edge. Higher threshold values result in less image edges being expressed. Lower threshold values result in more image edges being expressed.*Threshold –*If a pixel has been determined as forming part of an image edge, the resulting pixel colour will be determined by the colour value specified by the user.*Colour –*

The following image is a screenshot of the ** Stained Glass Image Filter** sample application in action:

*Locarno: Block Size 10, Factor 4, Euclidean*

### Stained Glass Image Filter

The ** Stained Glass Image Filter** detailed in this article operates on the basis of implementing modifications upon a specified sample/input image, producing resulting images which resemble the appearance of stained glass artwork.

A common variant of stained glass artwork comes in the form of several individual pieces of coloured glass being combined in order to create an image. The sample source code employs a similar method of combining what appears to be non-uniform puzzle pieces. The following list provides a broad overview of the steps involved in applying a ** Stained Glass Image Filter**:

Through rendering a Voronoi Diagram the resulting image will be divided into a number of regions. Each region being intended to represent an individual glass puzzle piece. The following section of this article provides a detailed discussion on rendering Voronoi Diagrams.*Render a Voronoi Diagram –*Each pixel forming part of the source/input image should be iterated. Whilst iterating pixels determine the region to which a pixel should be associated. A pixel should be associated to the region whose border has been determined the nearest to the pixel. In a following section of this article a detailed discussion regarding*Assign each Pixel to a Voronoi Diagram Region –*can be found.*Pixel Coordinate Distance Calculations*Each region will only express a single colour value. A region’s colour equates to the average colour as expressed by all the pixels forming part of a region. Once the average colour value of a region has been determined every pixel forming part of that region should be set to the average colour.*Determine each Region’s Colour Mean –*If the user configuration option indicates that edge detection should be implemented, apply*Implement Edge Detection –*. This method of edge detection has been discussed in detailed in a following section of this article.*Gradient Based Edge Detection*

*Bad Ragaz: Block Size 10, Factor 1, Manhattan*

### Voronoi Diagrams

Voronoi Diagrams represent a fairly uncomplicated concept. In contrast, the implementation of Voronoi Diagrams prove somewhat more of a challenge. From Wikipedia we gain the following definition:

In mathematics, a

Voronoi diagramis a way of dividing space into a number of regions. A set of points (called seeds, sites, or generators) is specified beforehand and for each seed there will be a corresponding region consisting of all points closer to that seed than to any other. The regions are called Voronoi cells. It is dual to the Delaunay triangulation.

In this article Voronoi Diagrams are generated resulting in regions expressing random shapes. Although region shapes are randomly generated, the parameters or ranges within which random values are selected are fixed/constant. The steps required in generating a Voronoi Diagram can be detailed as follows:

By making use of the user specified Block/Region Size value, group pixels together into square regions.*Define fixed size square regions –*Determine the sum total of pixel colour components of all the pixels forming part of a square region. The colour sum total value should be used as a seed value when generating random numbers in the next step.*Determine a Seed Value for Random number generation –***Determine a Random XY coordinate within each square region –**Generate two random numbers, specifying each region’s coordinate boundaries as minimum and maximum boundaries in generating random numbers. Keep record of every new randomly generated XY-Coordinate value.A pixel should be associated to the Random Coordinate point nearest to that pixel. Determine the Random Coordinate nearest to each pixel in the source/input image. The method implemented in calculating coordinate distance depends on the configuration value specified by the user.*Associate Pixels and Regions –*Each pixel forming part of the same region should be set to the same colour. The colour assigned to a region’s pixels will be determined by the average colour value of the region’s pixels.*Set Region Colours –*

The following image illustrates an example Voronoi Diagram consisting of 10 regions:

*Port Edward: Block Size 10, Factor 1, Chebyshev, Edge Threshold 2*

### Calculating Pixel Coordinate Distances

The sample source code provides three different coordinate distance calculation methods. The supported methods are: Euclidean, Manhattan and Chebyshev. A pixel’s nearest randomly generated coordinate depends on the distance between that pixel and the random coordinate. Each method of calculating distance in most instances would be likely to produce different output values, which in turn influences the region to which a pixel will be associated.

The most common method of distance calculation, Euclidean distance, has been described by Wikipedia as follows:

In mathematics, the

Euclidean distanceorEuclidean metricis the "ordinary" distance between two points that one would measure with a ruler, and is given by the Pythagorean formula. By using this formula as distance, Euclidean space (or even any inner product space) becomes a metric space. The associated norm is called theEuclidean norm.Older literature refers to the metric asPythagorean metric.

When calculating Euclidean distance the algorithm implemented can be expressed as follows:

*Zurich: Block Size 10, Factor 1, Euclidean*

As an alternative to calculating Euclidean distance, the sample source code also implements Manhattan Distance calculation. Often Manhattan Distance calculation will be referred to as City Block, Taxicab Geometry or rectilinear distance. From Wikipedia we gain the following description:

Taxicab geometry, considered by Hermann Minkowski in the 19th century, is a form of geometry in which the usual distance function or metric of Euclidean geometry is replaced by a new metric in which the distance between two points is the sum of the absolute differences of their coordinates. The taxicab metric is also known asrectilinear distance,orL_{1}distancenorm(seeL^{p}space),city block distance,Manhattan distance, orManhattan length, with corresponding variations in the name of the geometry.^{[1]}The latter names allude to the grid layout of most streets on the island of Manhattan, which causes the shortest path a car could take between two intersections in the borough to have length equal to the intersections’ distance in taxicab geometry

When calculating Manhattan Distance the algorithm implemented can be expressed as follows:

*Port Edward: Block Size 10, Factor 4, Euclidean*

Chebyshev Distance, a distance algorithm resembling the way in which a King Chess piece may move on a chess board. The following description we gain from Wikipedia:

In mathematics,

Chebyshev distance(orTchebychev distance),Maximum metric, or L∞ metric^{[1]}is a metric defined on a vector space where the distance between two vectors is the greatest of their differences along any coordinate dimension.^{[2]}It is named after Pafnuty Chebyshev.It is also known as

chessboard distance, since in the game of chess the minimum number of moves needed by a king to go from one square on a chessboard to another equals the Chebyshev distance between the centers of the squares, if the squares have side length one, as represented in 2-D spatial coordinates with axes aligned to the edges of the board.^{[3]}For example, the Chebyshev distance between f6 and e2 equals 4.

When calculating Chebyshev Distance the algorithm implemented can be expressed as follows:

*Salzburg: Block Size 20, Factor 1, Chebyshev*

### Gradient Based Edge Detection

Various methods of image edge detection can easily be implemented in C#. Each method of edge detection provides a set of benefits, usually weighed against a set of trade-offs. In this article and the accompanying sample source code the ** Gradient Based Edge Detection** method has been implement.

Take into regard that every region within the rendered Voronoi Diagram will only express a single colour, although most regions differ in the single colour they express. Once all pixels have been associated to a region and all pixel colour values have been updated the resulting image defines mostly clearly distinguishable colour gradients. A Gradient Based method of edge detection performs efficiently at detecting image edges. The edges detected are defined between different regions.

An image gradient can be considered as a difference in colour intensity relating to a specific direction. Only once all tasks related to applying the ** Stained Glass Filter** have been completed should the

**be applied. The steps involved in applying**

*Gradient Based Edge Detection***can be described as follows:**

*Gradient Based Edge Detection*Each pixel forming part of a source/input image should be iterated.*Iterate each pixel –*Calculate the colour value difference between the currently iterated pixel’s*Determine Horizontal and Vertical Gradients –*and*left*neighbour pixel as well as the*right*and*top*neighbour pixel. If the gradient exceeds the specified threshold continue to step 8.*bottom*Calculate the colour value difference between the currently iterated pixel’s*Determine Horizontal Gradient –*and*left*neighbour pixel. If the gradient exceeds the specified threshold continue to step 8.*right*Calculate the colour value difference between the currently iterated pixel’s*Determine Vertical Gradient –*and*top*neighbour pixel. If the gradient exceeds the specified threshold continue to step 8.*bottom*Calculate the colour value difference between the currently iterated pixel’s*Determine Diagonal Gradients –*and*North-Western*neighbour pixel as well as the**South-Eastern**and*North-Eastern*neighbour pixel. If the gradient exceeds the specified threshold continue to step 8.*South-Western*Calculate the colour value difference between the currently iterated pixel’s*Determine NW-SE Gradient –*and*North-Western*neighbour pixel. If the gradient exceeds the specified threshold continue to step 8.*South-Eastern*Calculate the colour value difference between the currently iterated pixel’s*Determine NE-SW Gradient –*and*North-Eastern*neighbour pixel.*South-Western*If any of the six gradients calculated exceeded the specified threshold value set the related pixel in the resulting image to the*Determine and set result pixel value –*specified by the user, if not, set the related pixel equal to the source pixel colour value.*Edge Colour*

*Zurich: Block Size 10, Factor 4, Chebyshev*

### Implementing a Stained Glass Image Filter

The sample source code defines two helper classes, both implemented when applying the ** Stained Glass Image Filter**. The

**class represents a single pixel in terms of an**

*Pixel***and**

*XY-Coordinate***,**

*Red***and**

*Green***values. The definition as follows:**

*Blue*public class Pixel { private int xOffset = 0; public int XOffset { get { return xOffset; } set { xOffset = value; } }

private int yOffset = 0; public int YOffset { get { return yOffset; } set { yOffset = value; } }

private byte blue = 0; public byte Blue { get { return blue; } set { blue = value; } }

private byte green = 0; public byte Green { get { return green; } set { green = value; } }

private byte red = 0; public byte Red { get { return red; } set { red = value; } } }

*Zurich: Block Size 10, Factor 1, Chebyshev, Edge Threshold 1*

The ** VoronoiPoint** class serves as method of recording randomly generated coordinates and referencing a region’s associated pixels. The definition as follows:

public class VoronoiPoint { private int xOffset = 0; public int XOffset { get { return xOffset; } set { xOffset = value; } }

private int yOffset = 0; public int YOffset { get { return yOffset; } set { yOffset = value; } }

private int blueTotal = 0; public int BlueTotal { get { return blueTotal; } set { blueTotal = value; } }

private int greenTotal = 0; public int GreenTotal { get {return greenTotal; } set { greenTotal = value; } }

private int redTotal = 0; public int RedTotal { get { return redTotal; } set { redTotal = value; } }

public void CalculateAverages() { if (pixelCollection.Count > 0) { blueAverage = blueTotal / pixelCollection.Count; greenAverage = greenTotal / pixelCollection.Count; redAverage = redTotal / pixelCollection.Count; } }

private int blueAverage = 0; public int BlueAverage { get { return blueAverage; } }

private int greenAverage = 0; public int GreenAverage { get { return greenAverage; } }

private int redAverage = 0; public int RedAverage { get { return redAverage; } }

private List<Pixel> pixelCollection = new List<Pixel>(); public List<Pixel> PixelCollection { get { return pixelCollection; } }

public void AddPixel(Pixel pixel) { blueTotal += pixel.Blue; greenTotal += pixel.Green; redTotal += pixel.Red;

pixelCollection.Add(pixel); } }

*Zurich: Block Size 20, Factor 1, Euclidean, Edge Threshold 1*

From the perspective of a filter implementation code base the only requirement comes in the form of having to invoke the ** StainedGlassColorFilter** extension method, no additional work is required from external code consumers. The

**method has been defined as an extension method targeting the Bitmap class. The**

*StainedGlassColorFilter***method definition as follows:**

*StainedGlassColorFilter*public static Bitmap StainedGlassColorFilter(this Bitmap sourceBitmap, int blockSize, double blockFactor, DistanceFormulaType distanceType, bool highlightEdges, byte edgeThreshold, Color edgeColor) { BitmapData sourceData = sourceBitmap.LockBits(new Rectangle(0, 0, sourceBitmap.Width, sourceBitmap.Height), ImageLockMode.ReadOnly, PixelFormat.Format32bppArgb);

byte[] pixelBuffer = new byte[sourceData.Stride * sourceData.Height];

byte[] resultBuffer = new byte[sourceData.Stride * sourceData.Height];

Marshal.Copy(sourceData.Scan0, pixelBuffer, 0, pixelBuffer.Length);

sourceBitmap.UnlockBits(sourceData);

int neighbourHoodTotal = 0; int sourceOffset = 0; int resultOffset = 0; int currentPixelDistance = 0; int nearestPixelDistance = 0; int nearesttPointIndex = 0;

Random randomizer = new Random();

List<VoronoiPoint> randomPointList = new List<VoronoiPoint>();

for (int row = 0; row < sourceBitmap.Height - blockSize; row += blockSize) { for (int col = 0; col < sourceBitmap.Width - blockSize; col += blockSize) { sourceOffset = row * sourceData.Stride + col * 4;

neighbourHoodTotal = 0;

for (int y = 0; y < blockSize; y++) { for (int x = 0; x < blockSize; x++) { resultOffset = sourceOffset + y * sourceData.Stride + x * 4; neighbourHoodTotal += pixelBuffer[resultOffset]; neighbourHoodTotal += pixelBuffer[resultOffset + 1]; neighbourHoodTotal += pixelBuffer[resultOffset + 2]; } }

randomizer = new Random(neighbourHoodTotal);

VoronoiPoint randomPoint = new VoronoiPoint(); randomPoint.XOffset = randomizer.Next(0, blockSize) + col; randomPoint.YOffset = randomizer.Next(0, blockSize) + row;

randomPointList.Add(randomPoint); } }

int rowOffset = 0; int colOffset = 0;

for (int bufferOffset = 0; bufferOffset < pixelBuffer.Length - 4; bufferOffset += 4) { rowOffset = bufferOffset / sourceData.Stride; colOffset = (bufferOffset % sourceData.Stride) / 4;

currentPixelDistance = 0; nearestPixelDistance = blockSize * 4; nearesttPointIndex = 0;

List<VoronoiPoint> pointSubset = new List<VoronoiPoint>();

pointSubset.AddRange(from t in randomPointList where rowOffset >= t.YOffset - blockSize * 2 && rowOffset <= t.YOffset + blockSize * 2 select t);

for (int k = 0; k < pointSubset.Count; k++) { if (distanceType == DistanceFormulaType.Euclidean) { currentPixelDistance = CalculateDistanceEuclidean(pointSubset[k].XOffset, colOffset, pointSubset[k].YOffset, rowOffset); } else if (distanceType == DistanceFormulaType.Manhattan) { currentPixelDistance = CalculateDistanceManhattan(pointSubset[k].XOffset, colOffset, pointSubset[k].YOffset, rowOffset); } else if (distanceType == DistanceFormulaType.Chebyshev) { currentPixelDistance = CalculateDistanceChebyshev(pointSubset[k].XOffset, colOffset, pointSubset[k].YOffset, rowOffset); } if (currentPixelDistance <= nearestPixelDistance) { nearestPixelDistance = currentPixelDistance; nearesttPointIndex = k; if (nearestPixelDistance <= blockSize / blockFactor) { break; } } }

Pixel tmpPixel = new Pixel (); tmpPixel.XOffset = colOffset; tmpPixel.YOffset = rowOffset; tmpPixel.Blue = pixelBuffer[bufferOffset]; tmpPixel.Green = pixelBuffer[bufferOffset + 1]; tmpPixel.Red = pixelBuffer[bufferOffset + 2];

pointSubset[nearesttPointIndex].AddPixel(tmpPixel); }

for (int k = 0; k < randomPointList.Count; k++) { randomPointList[k].CalculateAverages();

for (int i = 0; i < randomPointList[k].PixelCollection.Count; i++) { resultOffset = randomPointList[k].PixelCollection[i].YOffset * sourceData.Stride + randomPointList[k].PixelCollection[i].XOffset * 4;

resultBuffer[resultOffset] = (byte)randomPointList[k].BlueAverage; resultBuffer[resultOffset + 1] = (byte)randomPointList[k].GreenAverage; resultBuffer[resultOffset + 2] = (byte)randomPointList[k].RedAverage;

resultBuffer[resultOffset + 3] = 255; } }

Bitmap resultBitmap = new Bitmap(sourceBitmap.Width, sourceBitmap.Height);

BitmapData resultData = resultBitmap.LockBits(new Rectangle (0, 0, resultBitmap.Width, resultBitmap.Height), ImageLockMode.WriteOnly, PixelFormat.Format32bppArgb);

Marshal.Copy(resultBuffer, 0, resultData.Scan0, resultBuffer.Length);

resultBitmap.UnlockBits(resultData);

if (highlightEdges == true ) { resultBitmap = resultBitmap.GradientBasedEdgeDetectionFilter(edgeColor, edgeThreshold); }

return resultBitmap; }

*Locarno: Block Size 10, Factor 4, Euclidean, Edge Threshold 1*

### Implementing Pixel Coordinate Distance Calculations

As mentioned earlier, this article and the accompanying sample source code support coordinate distance calculations through three different calculation methods, namely Euclidean, Manhattan and Chebyshev. The method of distance calculation implemented depends on the configuration option specified by the user.

The ** CalculateDistanceEuclidean** method calculates distance implementing the Euclidean Distance Calculation method. In order to aid faster execution this method will calculate the square root of a specific value only once. Once a square root has been calculated the result is kept in memory. The following code snippet lists the definition of the

**method:**

*CalculateDistanceEuclidean*private static Dictionary <int,int> squareRoots = new Dictionary<int,int>();

private static int CalculateDistanceEuclidean(int x1, int x2, int y1, int y2) { int square = (x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2);

if(squareRoots.ContainsKey(square) == false) { squareRoots.Add(square, (int)Math.Sqrt(square)); }

return squareRoots[square]; }

The two other methods of calculating distance are implemented through the ** CalculateDistanceManhattan** and

**methods. The definition as follows:**

*CalculateDistanceChebyshev*private static int CalculateDistanceManhattan(int x1, int x2, int y1, int y2) { return Math.Abs(x1 - x2) + Math.Abs(y1 - y2); }

private static int CalculateDistanceChebyshev(int x1, int x2, int y1, int y2) { return Math.Max(Math.Abs(x1 - x2), Math.Abs(y1 - y2)); }

*Bad Ragaz: Block Size 12, Factor 1, Chebyshev*

### Implementing Gradient Based Edge Detection

Did you notice the very last step performed by the ** StainedGlassColorFilter** method involves implementing

**, depending on whether edge detection had been specified by the user.**

*Gradient Based Edge Detection*The following code snippet provides the implementation of the ** GradientBasedEdgeDetectionFilter** extension method:

public static Bitmap GradientBasedEdgeDetectionFilter( this Bitmap sourceBitmap, Color edgeColour, byte threshold = 0) { BitmapData sourceData = sourceBitmap.LockBits(new Rectangle (0, 0, sourceBitmap.Width, sourceBitmap.Height), ImageLockMode.ReadOnly, PixelFormat.Format32bppArgb);

byte[] pixelBuffer = new byte[sourceData.Stride * sourceData.Height]; byte[] resultBuffer = new byte[sourceData.Stride * sourceData.Height];

Marshal.Copy(sourceData.Scan0, pixelBuffer, 0, pixelBuffer.Length); Marshal.Copy(sourceData.Scan0, resultBuffer, 0, resultBuffer.Length);

sourceBitmap.UnlockBits(sourceData);

int sourceOffset = 0, gradientValue = 0; bool exceedsThreshold = false;

for(int offsetY = 1; offsetY < sourceBitmap.Height - 1; offsetY++) { for(int offsetX = 1; offsetX < sourceBitmap.Width - 1; offsetX++) { sourceOffset = offsetY * sourceData.Stride + offsetX * 4; gradientValue = 0; exceedsThreshold = true;

// Horizontal Gradient CheckThreshold(pixelBuffer, sourceOffset - 4, sourceOffset + 4, ref gradientValue, threshold, 2); // Vertical Gradient exceedsThreshold = CheckThreshold(pixelBuffer, sourceOffset - sourceData.Stride, sourceOffset + sourceData.Stride, ref gradientValue, threshold, 2);

if (exceedsThreshold == false) { gradientValue = 0;

// Horizontal Gradient exceedsThreshold = CheckThreshold(pixelBuffer, sourceOffset - 4, sourceOffset + 4, ref gradientValue, threshold);

if (exceedsThreshold == false) { gradientValue = 0;

// Vertical Gradient exceedsThreshold = CheckThreshold(pixelBuffer, sourceOffset - sourceData.Stride, sourceOffset + sourceData.Stride, ref gradientValue, threshold);

if (exceedsThreshold == false) { gradientValue = 0;

// Diagonal Gradient : NW-SE CheckThreshold(pixelBuffer, sourceOffset - 4 - sourceData.Stride, sourceOffset + 4 + sourceData.Stride, ref gradientValue, threshold, 2); // Diagonal Gradient : NE-SW exceedsThreshold = CheckThreshold(pixelBuffer, sourceOffset - sourceData.Stride + 4, sourceOffset - 4 + sourceData.Stride, ref gradientValue, threshold, 2);

if (exceedsThreshold == false) { gradientValue = 0;

// Diagonal Gradient : NW-SE exceedsThreshold = CheckThreshold(pixelBuffer, sourceOffset - 4 - sourceData.Stride, sourceOffset + 4 + sourceData.Stride, ref gradientValue, threshold);

if (exceedsThreshold == false) { gradientValue = 0;

// Diagonal Gradient : NE-SW exceedsThreshold = CheckThreshold(pixelBuffer, sourceOffset - sourceData.Stride + 4, sourceOffset + sourceData.Stride - 4, ref gradientValue, threshold); } } } } }

if (exceedsThreshold == true) { resultBuffer[sourceOffset] = edgeColour.B; resultBuffer[sourceOffset + 1] = edgeColour.G; resultBuffer[sourceOffset + 2] = edgeColour.R; }

resultBuffer[sourceOffset + 3] = 255; } }

Bitmap resultBitmap = new Bitmap (sourceBitmap.Width, sourceBitmap.Height); BitmapData resultData = resultBitmap.LockBits(new Rectangle (0, 0, resultBitmap.Width, resultBitmap.Height), ImageLockMode .WriteOnly, PixelFormat.Format32bppArgb);

Marshal.Copy(resultBuffer, 0, resultData.Scan0, resultBuffer.Length); resultBitmap.UnlockBits(resultData);

return resultBitmap; }

*Zurich: Block Size 15, Factor 1, Manhattan, Edge Threshold 1*

### Sample Images

This article features a rendered graphic illustrating an example Voronoi Diagram which has been released into the **public domain** by its author, **Augochy at the wikipedia project**. This applies worldwide. The original can be downloaded from Wikipedia.

All of the photos that appear in this article were taken by myself. Photos listed under ** Zurich**,

**and**

*Locarno***were shot in**

*Bad Ragaz***. The photo listed as**

*Switzerland***had been shot in**

*Salzburg***and the photo listed under**

*Austria***had been shot in**

*Port Edward***. In order to fully realize the extent to which images had been modified the following section details the original photos.**

*South Africa**Zurich, Switzerland*

*Salzburg, Austria*

*Locarno, Switzerland*

*Bad Ragaz, Switzerland*

*Port Edward, South Africa*

*Zurich, Switzerland*

*Zurich, Switzerland*

*Zurich, Switzerland*

*Bad Ragaz, Switzerland*

### Related Articles and Feedback

Feedback and questions are always encouraged. If you know of an alternative implementation or have ideas on a more efficient implementation please share in the comments section.

I’ve published a number of articles related to imaging and images of which you can find URL links here:

- C# How to: Image filtering by directly manipulating Pixel ARGB values
- C# How to: Image filtering implemented using a ColorMatrix
- C# How to: Blending Bitmap images using colour filters
- C# How to: Bitmap Colour Substitution implementing thresholds
- C# How to: Generating Icons from Images
- C# How to: Swapping Bitmap ARGB Colour Channels
- C# How to: Bitmap Pixel manipulation using LINQ Queries
- C# How to: Linq to Bitmaps – Partial Colour Inversion
- C# How to: Bitmap Colour Balance
- C# How to: Bi-tonal Bitmaps
- C# How to: Bitmap Colour Tint
- C# How to: Bitmap Colour Shading
- C# How to: Image Solarise
- C# How to: Image Contrast
- C# How to: Bitwise Bitmap Blending
- C# How to: Image Arithmetic
- C# How to: Image Convolution
- C# How to: Image Edge Detection
- C# How to: Difference Of Gaussians
- C# How to: Image Median Filter
- C# How to: Image Unsharp Mask
- C# How to: Image Colour Average
- C# How to: Image Erosion and Dilation
- C# How to: Morphological Edge Detection
- C# How to: Boolean Edge Detection
- C# How to: Gradient Based Edge Detection
- C# How to: Sharpen Edge Detection
- C# How to: Calculating Gaussian Kernels
- C# How to: Image Blur
- C# How to: Image Transform Rotate
- C# How to: Image Transform Shear
- C# How to: Compass Edge Detection
- C# How to: Oil Painting and Cartoon Filter

## 26 Responses to “C# How to: Stained Glass Image Filter”