I've developing the robot which can run in the corn plant and guided by compass sensors but I want to apply the camera as the eye of robot and use the image processing to detect error angle of movement.
This is the image examples.
processed image raw image segmented image
I use this following step
Step 1 : The current technique that I use is convert color value to HSV modified from this code
Step 2 : So It will detect selected color which is the brown or dirt color then I collect the most left and right of brown or selected color of each image row in two array(a red point).
- Step 3 : I plot a 2 linear regression line as a blue point and calculate intersection point as a pink point
- Step 4 : Draw the green line to compare the pink point with another images. I'm not sure what to do with this green line yet
- The problem is the dirt or brown color is exist between the corn leaf too then I make my code to miss a calculation
The question is how to filter out the brown pixel that is between the corn leaf or another area that is not in the corn path? Which algorithm or methodology that I should study or apply in this problem?
EDIT1 : Using Spektre's answer and It looks better
Here is the result after I apply it with JAVA+Boofcv
More Information
For your source image
I would:
create mask of brown stuff
Just treshold
H,S
and ignoreV
, You already have this. I use integer255
instead of color (Blue) for latter computation.blur the mask
This will remove small clusters of wrongly selected parts. After this you should treshold the mask again because the mask value will be a bit smaller then
255
unless you got fully selected areas. The bigger the area the bigger the value (closer to255
). I treshold with>=150
scan the mask by horizontal lines
for each line find the center of gravity of all selected pixels
After blurring and tresholding again the used mask is in Aqua. So compute the average point
x
coordinate of all masked pixels in each line. This point is marked with White.regress line through all of the centers of gravity
I use mine approximation search for this but you can use any regression you want. The regressed line is marked with Red
I used line equation
x=x0+y*dx
wherey=<0,pic1.ys>
. And search the solution on intervals:Where
pic1.xs,pic1.ys
is the image resolution. So as you can see I do not cover full range of angles but I think this would not work on those edge cases anyway (near horizontal directions). For such cases you should do this on the Vertical lines instead and usex=y0+x*dy
instead.Here C++ source I did this with:
I use my own
picture
class for images so some members are:xs,ys
size of image in pixelsp[y][x].dd
is pixel at(x,y)
position as 32 bit integer typep[y][x].dw[2]
is pixel at(x,y)
position as 2x16 bit integer type for 2D fieldsp[y][x].db[4]
is pixel at(x,y)
position as 4x8 bit integer type for easy channel accessclear(color)
- clears entire imageresize(xs,ys)
- resizes image to new resolutionbmp
- VCL encapsulated GDI Bitmap with Canvas accesssmooth(n)
- fast blur the imagen
timesYou can further improve this with segmentation (removing small clusters) based on area and position. Also you can ignore too big peaks in avg points between neighbors. Also you can detect sky and ignore whole area where sky is present.
[edit1] smooth
This is how my smooth looks like:
It simply weight average 3 pixels
and after that do the same with
to avoid shifting of image. Then this whole thing is looped
n
times and that is all. You can ignore the clamp and pixel-format options in this case it ispf==_pf_rgba
but it uses only single channel anyway ...dec_color,enc_color
just unpack,pack color channels into/from array of variables to avoid truncating and overflow issues on 8 bit channels and also to format/simplify the code a bit better (for different pixel format support)btw smooth base is the same as convolution with
and
If I am right you're asking about the brown portions which are seen astray or in other portions of the background?
How did you get that last image? I assume you multiplied the original image by a mask? Even if you didnt, you can simply get the mask from the image by selecting wherever the image exists (any simple, very low threshold would do). (Apply adaptive thresholding ,an even more accurate version of the original to get a better mask)
Take that mask and clean it up using morphological operations, in your case closing would suffice. Morphology is made up of a plethora of operations which can give you an extremely clean image mask. Read up on them.