Introduction
So in part one we decided to use the AForge.Net framework to do this: Finding the fiber and determining the average color based on the image. And the saga continues.
The average color
After talking to Andrew Kirillov the developer behind AForge.Net he told me about an easier way to do what I did.
And here it is.
Dim i = New Aforge.Imaging.Filters.ColorFiltering
Dim b = GetBitmap()
i.red = New intrange(150,255)
i.Green = New intrange(150,255)
i.blue = New intrange(150,255)
i.FillOutsideRange = False
Try
PictureBox2.Image = i.Apply(b)
Catch ex As Exception
MessageBox.Show(ex.Message)
End Try
b = GetBitmap()
Dim i2 = New AForge.Imaging.BlobCounter(b)
Try
Dim Biggestblob = i2.GetObjects(b,true).Where(Function(x) x.Area > 1000).Reverse().FirstOrDefault
PictureBox2.Image = i2.GetObjects(b,true).Where(Function(x) x.Area > 1000)(0).Image
Me.lblStatus.Text = "Mean: " & Biggestblob.ColorMean.ToString & " STD: " & Biggestblob.ColorStdDev.ToString
Me.lblaveragecolor.BackColor = Biggestblob.ColorMean
Me.HsbColorIndicator1.AverageColor = Biggestblob.ColorMean
Me.lblstdcolor.BackColor = Biggestblob.ColorStdDev
Catch ex As Exception
MessageBox.Show(ex.Message)
End Try```
This method is extremely fast. The result is the same.
Here is my version.
<div class="image_block">
<a href="https://lessthandot.z19.web.core.windows.net/wp-content/uploads/users/chrissie1/Findfiber/FindFiber2.png?mtime=1294139316"><img alt="" src="https://lessthandot.z19.web.core.windows.net/wp-content/uploads/users/chrissie1/Findfiber/FindFiber3.png?mtime=1294139316" width="753" height="544" /></a>
</div>
And here is the new method.
<div class="image_block">
<a href="https://lessthandot.z19.web.core.windows.net/wp-content/uploads/users/chrissie1/Findfiber/FindFiber3.png?mtime=1294139356"><img alt="" src="https://lessthandot.z19.web.core.windows.net/wp-content/uploads/users/chrissie1/Findfiber/FindFiber4.png?mtime=1294139356" width="753" height="544" /></a>
</div>
## Naming the color
And now that we have the average color of our fiber I want to name it. Since it is easier to talk about something when it has a name then when you have to site the RGB values. I thought about this mast night and while chatting with George Mastros he made me think about the HSB system and especially the H in that system. The H or hue makes it easy to separate colors. And it goes from 0 to 360. So I made a usercontrol to show the H values. Like this.
```vbnet
Imports System.Drawing.Drawing2D
Public Class HSBColorIndicator
Private _AverageColor As Color
Public Sub New()
InitializeComponent()
Dim rect = New Rectangle(0, 0, 20, 360)
Dim region = New Rectangle(0, -1, 18, 361)
Dim m_colorSliderBitmap = New Bitmap(18, 360)
Using g = Graphics.FromImage(m_colorSliderBitmap)
Using brBrush = New LinearGradientBrush(rect, Color.Blue, Color.Red, 90.0F, False)
Dim colorArray = {Color.Red, Color.Magenta, Color.Blue, Color.Cyan, Color.FromArgb(0, 255, 0), Color.Yellow, Color.Red}
Dim posArray = {0.0F, 0.1667F, 0.3372F, 0.502F, 0.6686F, 0.8313F, 1.0F}
Dim colorBlend = New ColorBlend()
colorBlend.Colors = colorArray
colorBlend.Positions = posArray
brBrush.InterpolationColors = colorBlend
g.FillRectangle(brBrush, region)
g.DrawLine(Pens.Black, 0, 30, 18, 30)
g.DrawLine(Pens.Black, 0, 90, 18, 90)
g.DrawLine(Pens.Black, 0, 150, 18, 150)
g.DrawLine(Pens.Black, 0, 210, 18, 210)
g.DrawLine(Pens.Black, 0, 270, 18, 270)
g.DrawLine(Pens.Black, 0, 330, 18, 330)
End Using
End Using
Me.hsbcolorbar.Image = m_colorSliderBitmap
End Sub
Public Property AverageColor As Color
Get
Return _AverageColor
End Get
Set(ByVal value As Color)
_AverageColor = value
DrawArrow
End Set
End Property
Private Sub DrawArrow()
Me.Invalidate(New Rectangle(19, 0, 200, 360))
Me.Refresh()
Dim x = 19
Dim y = Convert.ToInt32(360 - _AverageColor.GetHue)
Using g = Me.CreateGraphics
Dim points = {New Point(x, y), New Point(x + 10, y - 3), New Point(x + 10, y + 3)}
g.DrawPolygon(Pens.White, points)
g.DrawString(GetColorName, New Font(New FontFamily("Arial"), 7), Brushes.White, 31, y - 5)
End Using
End Sub
Private Function GetColorName() As String
Dim _Color As String = "Not defined"
Dim hue = 360 - _AverageColor.GetHue
If hue > 30 AndAlso hue <= 90 Then
_Color = "Violet"
ElseIf hue > 90 AndAlso hue <= 150 Then
_Color = "Blue"
ElseIf hue > 150 AndAlso hue <= 210 Then
_Color = "Turquoise"
ElseIf hue > 210 AndAlso hue <= 270 Then
_Color = "Green"
ElseIf hue > 270 AndAlso hue <= 330 Then
_Color = "Yellow"
Else
_Color = "Red"
End If
Return _Color
End Function
End Class
So I decided on some boundaries and I show an arrow where the average color is situated within the hue range, simple. Getting the hue from a color in .Net is very easy since you can just do GetHue().
And here is the result of the Belgian jury.
And I’m actually quit pleased with the result.
Conclusion
Working with colors is not that hard but you need to know about a lot of things to make it easy and manageable. Every color system has it uses and it’s advantages and disadvantages.