Region coloring is to put a unique label to a separated component. At first it labels all the pixels in a way and at a second phase, those labels are merged when connected. The result of the region coloring is still a area image with each component assigned with a unique number. And if you want to count how many defects are there, the area has to be scanned and count unique number. Or if you want to transmit the defect information, the whole labeled image has to be sent out - though the size can be much smaller than original image if run length encoding is used.
Contour detection is to trace the defect pixels and extract contour only. This start on a pixel and keep following on nearby defect pixels until it comes back to the start pixel. The result of this segmentation is only contour pixels and can apply chain coding which takes 3 bits per pixel. And it makes quite small size and compact to transmit to other. This contour information can be used to calculate area or moment for analysis.
There are number of algorithms for contour tracing. Refer [CTA]. The square tracing doesn't work with 8 neighbor hood. The Moore-Neighbor tracing and radial sweep algorithms are almost identical. Theo Pavlidis's algorithm is not that easy to implement.
In here, I will show implementation of Moore-Neighbor Tracing based on the above web site. Though the stop condition at the web site is not so simple and will use approach from other book - P192 of [IPA] as below exerpt.
If the current boundary element Pn is equal to the second border element P1 , and if the previous border element Pn-1 is equal to Po , stop. Otherwise repeat step 2.First the image has to be traversed to find the start point. And when found a pixel, do the contour tracing and clear out the traced object so that it does not get picked up again. Here is the code snippet.
void Build( BinaryImageBase& image, const Region& roi, PointsContainer* contours ) { for( int sy = roi.Top; sy<= roi.Bottom; ++sy ) { for( int sx = roi.Left; sx<=roi.Right; ++sx ) { if( image.GetPixel( sx, sy ) == false ) continue; Points contour; m_Tracer->Trace( image, roi, Point((uint16_t)sx,(uint16_t)sy), &contour ); // MooreNeighbor Tracing m_Filler->Fill( image, contour[0], false ); // FloodFill contours->push_back( contour ); } } }The Moore-Neighbor tracing can be implemented as below.
struct WalkTableItem { Point Step; int Next; } WalkTable[8] = { { Point(1,0), 6, }, // from North to North-East, when found pixel, start from index 6 of this table { Point(0,1), 0, }, // from North-East to East { Point(0,1), 0, }, // from East to South-East { Point(-1,0), 2, }, // from South-East to South { Point(-1,0), 2, }, // from South to South-West { Point(0,-1), 4, }, // from South-West to West { Point(0,-1), 4, }, // from West to North-West { Point(1,0), 6, }, // from North-West to North }; void Trace( const BinaryImageBase& image, const Region& roi, const Point& start, Points* trace ) { const int SizeOfT = sizeof(WalkTable)/sizeof(WalkTable[0]); WalkTableItem (&T)[SizeOfT]( WalkTable ); // access WalkTable with variable name T for short Points B; B.push_back( start ); // B is the container of points int sk = 0, k = 0; // sk is the start index, k is current index Point c = start - T[k].Step; // c is current point k = T[k].Next; sk = ( SizeOfT + k - 1 ) % SizeOfT; for( ; ; ) { Point n = c + T[k].Step; // n is next point if( image.GetPixel( n.As.XY.X, n.As.XY.Y ) == true ) { size_t bc = B.size(); if( bc > 2 && n == B[1] && B[bc-1] == B[0] ) { break; // end condition by [IPA] } else { B.push_back( n ); // found contour pixel n = c; // back-track k = T[k].Next; // next index to search sk = ( SizeOfT + k - 1 ) % SizeOfT; // stop index if no pixel at moore-neighbor } } else { k = (k+1) % SizeOfT; // move to next index } c = n; if( sk == k ) // stop condition with 1 pixel defect { break; } } if( B.size() == 1 ) { trace->swap( B ); // 1 pixel defect } else { trace->swap( Points( &B[0], &B[ B.size() - 1 ] ) ); // The last pixel is the start pixel so exclude it. } }Here is code of stack-based FloodFill based on [WFF].
void Fill( BinaryImageBase& image, const Point& start, bool fillValue ) { std::stack< Point > Q; Q.push( start ); while( Q.size() != 0 ) { Point p = Q.top(); Q.pop(); if( p.As.XY.X < 0 || p.As.XY.X >= image.GetWidth() || p.As.XY.Y < 0 || p.As.XY.Y >= image.GetHeight() ) continue; if( image.GetPixel( p.As.XY.X, p.As.XY.Y ) == fillValue ) continue; image.SetPixel( p.As.XY.X, p.As.XY.Y, fillValue ); Q.push( Point( p.As.XY.X + 1, p.As.XY.Y ) ); Q.push( Point( p.As.XY.X - 1, p.As.XY.Y ) ); Q.push( Point( p.As.XY.X + 1, p.As.XY.Y - 1 ) ); Q.push( Point( p.As.XY.X, p.As.XY.Y - 1 ) ); Q.push( Point( p.As.XY.X - 1, p.As.XY.Y - 1 ) ); Q.push( Point( p.As.XY.X + 1, p.As.XY.Y + 1 ) ); Q.push( Point( p.As.XY.X, p.As.XY.Y + 1 ) ); Q.push( Point( p.As.XY.X - 1, p.As.XY.Y + 1 ) ); } }
The result of contour tracing is boundary pixels and one or more pixels can be repeated as it needs to return to the original image. Refer below image and it's result.
Contour pixels : (1,1),(2,2),(3,1),(2,2),(3,3),(2,2)
Reference :
[CTA] Contour Tracing WEB Site by Abeer George Ghuneim
[IPA] Image Processing, Analysis and Machine Vision by Milan Sonka, Vaclav Hlavac, Roger Boyle.
[WFF] FloodFill from Wikipedia
No comments:
Post a Comment