All Downloads are FREE. Search and download functionalities are using the official Maven repository.

boofcv.alg.shapes.ellipse.BinaryEllipseDetectorPixel Maven / Gradle / Ivy

Go to download

BoofCV is an open source Java library for real-time computer vision and robotics applications.

There is a newer version: 1.1.7
Show newest version
/*
 * Copyright (c) 2011-2019, Peter Abeles. All Rights Reserved.
 *
 * This file is part of BoofCV (http://boofcv.org).
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package boofcv.alg.shapes.ellipse;

import boofcv.abst.filter.binary.BinaryLabelContourFinder;
import boofcv.alg.filter.binary.ContourPacked;
import boofcv.factory.filter.binary.FactoryBinaryContourFinder;
import boofcv.struct.ConnectRule;
import boofcv.struct.distort.PixelTransform;
import boofcv.struct.image.GrayS32;
import boofcv.struct.image.GrayU8;
import georegression.fitting.curves.ClosestPointEllipseAngle_F64;
import georegression.fitting.curves.FitEllipseAlgebraic_F64;
import georegression.geometry.UtilEllipse_F64;
import georegression.struct.curve.EllipseQuadratic_F64;
import georegression.struct.curve.EllipseRotated_F64;
import georegression.struct.point.Point2D_F32;
import georegression.struct.point.Point2D_F64;
import georegression.struct.point.Point2D_I32;
import org.ddogleg.struct.FastQueue;

import java.util.List;

/**
 * 

Detects ellipses inside a binary image by finding their contour and fitting an ellipse to the contour. Fitting * is done using pixel precise contour points. See {@link SnapToEllipseEdge} for a way to use sub-pixel points * and improve the fit's accuracy. Optionally, lens distortion can be removed from the contour points prior * to processing.

* * After the contour has been found an ellipse is fit to them using the {@link FitEllipseAlgebraic_F64 algebraic} * formulation. Points which are not approximately ellipsoidal are removed. * Approximately ellipsoidal is defined by the distance of the farthest contour point away from the ellipse. For * computational efficiency reasons a maximum of 20 points are sampled. If there are more than 20 points in * the contour then they are evenly sampled across the contour. Only external contours are considered. * * Parameters: *
*
maxDistanceFromEllipse
*
maximum distance from the ellipse in pixels
*
minimumContour
*
minimum number of pixels in the contour
*
maximumContour
*
maximum number of pixels in the contour
*
internalContour
*
If true it will process internal contours
*
* * @author Peter Abeles */ public class BinaryEllipseDetectorPixel { // maximum distance from the ellipse in pixels private double maxDistanceFromEllipse = 3.0; // minimum number of pixels in the contour private int minimumContour = 20; // maximum number of pixels in the contour. 0 == no limit private int maximumContour = 0; // minimum number of pixels along the minor axis private double minimumMinorAxis = 1.5; // Can be used to filter out shapes which are very skinny private double maxMajorToMinorRatio = Double.MAX_VALUE; private boolean internalContour = false; private BinaryLabelContourFinder contourFinder; private GrayS32 labeled = new GrayS32(1,1); private FitEllipseAlgebraic_F64 algebraic = new FitEllipseAlgebraic_F64(); private ClosestPointEllipseAngle_F64 closestPoint = new ClosestPointEllipseAngle_F64(1e-4f,15); // transforms which can be used to handle lens distortion protected PixelTransform distToUndist; protected Point2D_F32 distortedPoint = new Point2D_F32(); private boolean verbose = false; private FastQueue pointsF = new FastQueue<>(Point2D_F64.class, true); private FastQueue found = new FastQueue<>(Found.class, true); // temporary storage for a contour private FastQueue contourTmp = new FastQueue<>(Point2D_I32.class,true); public BinaryEllipseDetectorPixel(ConnectRule connectRule ) { contourFinder = FactoryBinaryContourFinder.linearChang2004(); contourFinder.setConnectRule(connectRule); } public BinaryEllipseDetectorPixel() { this(ConnectRule.FOUR); } /** *

Specifies transforms which can be used to change coordinates from distorted to undistorted. * The undistorted image is never explicitly created.

* *

* WARNING: The undistorted image must have the same bounds as the distorted input image. This is because * several of the bounds checks use the image shape. This are simplified greatly by this assumption. *

* * @param distToUndist Transform from distorted to undistorted image. */ public void setLensDistortion( PixelTransform distToUndist ) { this.distToUndist = distToUndist; } /** * Finds all valid ellipses in the binary image * @param binary binary image */ public void process( GrayU8 binary ) { found.reset(); labeled.reshape(binary.width, binary.height); contourFinder.process(binary, labeled); List blobs = contourFinder.getContours(); for (int i = 0; i < blobs.size(); i++) { ContourPacked c = blobs.get(i); contourFinder.loadContour(c.externalIndex,contourTmp); proccessContour(contourTmp.toList()); if(internalContour) { for( int j = 0; j < c.internalIndexes.size(); j++ ) { contourFinder.loadContour(c.internalIndexes.get(j),contourTmp); proccessContour(contourTmp.toList()); } } } } private void proccessContour(List contour) { if (contour.size() < minimumContour || (maximumContour > 0 && contour.size() > maximumContour) ) { if( verbose ) System.out.println("Rejecting: too small (or large) "+contour.size()); return; } // discard shapes which touch the image border if( touchesBorder(contour) ) return; pointsF.reset(); undistortContour(contour,pointsF); // fit it to an ellipse. This will just be approximate. The more precise technique is much slower if( !algebraic.process(pointsF.toList())) { if( verbose ) System.out.println("Rejecting: algebraic fit failed. size = "+pointsF.size()); return; } EllipseQuadratic_F64 quad = algebraic.getEllipse(); Found f = found.grow(); UtilEllipse_F64.convert(quad,f.ellipse); boolean accepted = true; if( f.ellipse.b <= minimumMinorAxis ) { if( verbose ) System.out.println("Rejecting: Minor axis too small. size = "+f.ellipse.b); accepted = false; } else if( !isApproximatelyElliptical(f.ellipse,pointsF.toList(),20)) { if( verbose ) System.out.println("Rejecting: Not approximately elliptical. size = "+pointsF.size()); accepted = false; } else if( f.ellipse.a > maxMajorToMinorRatio*f.ellipse.b ) { if( verbose ) System.out.println("Rejecting: Major to minor axis length ratio too extreme = "+pointsF.size()); accepted = false; } if( accepted ) { if (verbose) System.out.println("Success! size = " + pointsF.size()); adjustElipseForBinaryBias(f.ellipse); f.contour = contour; } else { found.removeTail(); } } /** * In a binary image the contour on the right and bottom is off by one pixel. This is because the block region * extends the entire pixel not just the lower extent which is where it is indexed from. */ protected void adjustElipseForBinaryBias( EllipseRotated_F64 ellipse ) { ellipse.center.x += 0.5; ellipse.center.y += 0.5; ellipse.a += 0.5; ellipse.b += 0.5; } protected final boolean touchesBorder( List contour ) { int endX = labeled.width-1; int endY = labeled.height-1; for (int j = 0; j < contour.size(); j++) { Point2D_I32 p = contour.get(j); if( p.x == 0 || p.y == 0 || p.x == endX || p.y == endY ) { return true; } } return false; } /** * Undistort the contour points and convert into a floating point format for the fitting operation * * @param external The external contour * @param pointsF Output of converted points */ void undistortContour(List external, FastQueue pointsF ) { for (int j = 0; j < external.size(); j++) { Point2D_I32 p = external.get(j); if( distToUndist != null ) { distToUndist.compute(p.x,p.y,distortedPoint); pointsF.grow().set( distortedPoint.x , distortedPoint.y ); } else { pointsF.grow().set(p.x, p.y); } } } /** * Look at the maximum distance contour points are from the ellipse and see if they exceed a maximum threshold */ boolean isApproximatelyElliptical(EllipseRotated_F64 ellipse , List points , int maxSamples ) { closestPoint.setEllipse(ellipse); double maxDistance2 = maxDistanceFromEllipse*maxDistanceFromEllipse; if( points.size() <= maxSamples ) { for( int i = 0; i < points.size(); i++ ) { Point2D_F64 p = points.get(i); closestPoint.process(p); double d = closestPoint.getClosest().distance2(p); if( d > maxDistance2 ) { return false; } } } else { for (int i = 0; i < maxSamples; i++) { Point2D_F64 p = points.get( i*points.size()/maxSamples ); closestPoint.process(p); double d = closestPoint.getClosest().distance2(p); if( d > maxDistance2 ) { return false; } } } return true; } public BinaryLabelContourFinder getContourFinder() { return contourFinder; } public boolean isVerbose() { return verbose; } public boolean isInternalContour() { return internalContour; } public void setInternalContour(boolean internalContour) { this.internalContour = internalContour; } public void setVerbose(boolean verbose) { this.verbose = verbose; } public double getMaxDistanceFromEllipse() { return maxDistanceFromEllipse; } public void setMaxDistanceFromEllipse(double maxDistanceFromEllipse) { this.maxDistanceFromEllipse = maxDistanceFromEllipse; } public int getMinimumContour() { return minimumContour; } public void setMinimumContour(int minimumContour) { this.minimumContour = minimumContour; } public int getMaximumContour() { return maximumContour; } public void setMaximumContour(int maximumContour) { this.maximumContour = maximumContour; } public double getMaxMajorToMinorRatio() { return maxMajorToMinorRatio; } public void setMaxMajorToMinorRatio(double maxMajorToMinorRatio) { this.maxMajorToMinorRatio = maxMajorToMinorRatio; } public double getMinimumMinorAxis() { return minimumMinorAxis; } public void setMinimumMinorAxis(double minimumMinorAxis) { this.minimumMinorAxis = minimumMinorAxis; } public List getFound() { return found.toList(); } public static class Found { /** * Computed ellipse in undistorted pixel coordinates */ public EllipseRotated_F64 ellipse = new EllipseRotated_F64(); /** * Contour in distorted pixel coordinates */ public List contour; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy