boofcv.alg.shapes.ellipse.BinaryEllipseDetector Maven / Gradle / Ivy
/*
* Copyright (c) 2011-2020, 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.alg.filter.binary.ContourPacked;
import boofcv.struct.distort.PixelTransform;
import boofcv.struct.image.GrayU8;
import boofcv.struct.image.ImageGray;
import georegression.struct.curve.EllipseRotated_F64;
import georegression.struct.point.Point2D_F32;
import georegression.struct.point.Point2D_I32;
import org.ddogleg.struct.FastQueue;
import java.util.ArrayList;
import java.util.List;
/**
* Detects ellipses inside gray scale images. The first step is to detect them in a binary image and then
* refine the fit using the gray scale image. Optionally the refinement step can be turned off or postponed
* until the user invokes it directly. False positives are pruned using the edge intensity check. This check removes
* ellipses with edges that are low intensity sicne they are most likely generated from noise.
*
* @author Peter Abeles
*/
public class BinaryEllipseDetector> {
BinaryEllipseDetectorPixel ellipseDetector;
SnapToEllipseEdge ellipseRefiner;
EdgeIntensityEllipse intensityCheck;
// storage for the output refined ellipses
FastQueue results = new FastQueue<>(EllipseInfo::new);
Class inputType;
boolean verbose = false;
// toggled the refinement step. If false an ellise can still be refined after the fact
boolean autoRefine = true;
/**
* Configures the detector
*
* @param ellipseDetector Detector which uses pixel precise edges
* @param ellipseRefiner Sub pixel edge refinement. If null the refinement step is skipped
* @param intensityCheck Computes the intensity of the edge to remove false positives
* @param inputType Input image type
*/
public BinaryEllipseDetector(BinaryEllipseDetectorPixel ellipseDetector,
SnapToEllipseEdge ellipseRefiner,
EdgeIntensityEllipse intensityCheck ,
Class inputType ) {
this.ellipseDetector = ellipseDetector;
this.ellipseRefiner = ellipseRefiner;
this.intensityCheck = intensityCheck;
this.inputType = inputType;
}
/**
* 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.
* @param undistToDist Transform from undistorted to distorted image.
*/
public void setLensDistortion(PixelTransform distToUndist , PixelTransform undistToDist ) {
this.ellipseDetector.setLensDistortion(distToUndist);
if( this.ellipseRefiner != null )
this.ellipseRefiner.setTransform(undistToDist);
this.intensityCheck.setTransform(undistToDist);
}
/**
* Detects ellipses inside the binary image and refines the edges for all detections inside the gray image
*
* @param gray Grayscale image
* @param binary Binary image of grayscale. 1 = ellipse and 0 = ignored background
*/
public void process(T gray, GrayU8 binary) {
results.reset();
ellipseDetector.process(binary);
if( ellipseRefiner != null)
ellipseRefiner.setImage(gray);
intensityCheck.setImage(gray);
List found = ellipseDetector.getFound();
for( BinaryEllipseDetectorPixel.Found f : found ) {
if( !intensityCheck.process(f.ellipse) ) {
if( verbose ) System.out.println("Rejecting ellipse. Initial fit didn't have intense enough edge");
continue;
}
EllipseInfo r = results.grow();
r.contour = f.contour;
if( ellipseRefiner != null ) {
if (!ellipseRefiner.process(f.ellipse, r.ellipse)) {
if( verbose ) System.out.println("Rejecting ellipse. Refined fit didn't have an intense enough edge");
results.removeTail();
continue;
} else if( !intensityCheck.process(f.ellipse) ) {
if( verbose ) System.out.println("Rejecting ellipse. Refined fit didn't have an intense enough edge");
continue;
}
} else {
r.ellipse.set(f.ellipse);
}
r.averageInside = intensityCheck.averageInside;
r.averageOutside = intensityCheck.averageOutside;
}
}
/**
* If auto refine is turned off an ellipse can be refined after the fact using this function, provided
* that the refinement algorithm was passed in to the constructor
* @param ellipse The ellipse to be refined
* @return true if refine was successful or false if not
*/
public boolean refine( EllipseRotated_F64 ellipse ) {
if( autoRefine )
throw new IllegalArgumentException("Autorefine is true, no need to refine again");
if( ellipseRefiner == null )
throw new IllegalArgumentException("Refiner has not been passed in");
if (!ellipseRefiner.process(ellipse,ellipse)) {
return false;
} else {
return true;
}
}
public BinaryEllipseDetectorPixel getEllipseDetector() {
return ellipseDetector;
}
public boolean isVerbose() {
return verbose;
}
public void setVerbose(boolean verbose) {
this.verbose = verbose;
}
public boolean isAutoRefine() {
return autoRefine;
}
public void setAutoRefine(boolean autoRefine) {
this.autoRefine = autoRefine;
}
public Class getInputType() {
return inputType;
}
public List getAllContours() {
return ellipseDetector.getContours();
}
/**
* Returns all the found ellipses in the input image.
*
* WARNING: Returned data is recycled on the next call to process
*
* @return List of found ellipses.
*/
public FastQueue getFound() {
return results;
}
public List getFoundEllipses( List storage ) {
if( storage == null )
storage = new ArrayList<>();
for (int i = 0; i < results.size; i++) {
storage.add( results.get(i).ellipse );
}
return storage;
}
public static class EllipseInfo {
public EllipseRotated_F64 ellipse = new EllipseRotated_F64();
public List contour;
/**
* Average pixel intensity along the inside of the fitted ellipse
*/
public double averageInside;
/**
* Average pixel intensity along the outside of the fitted ellipse
*/
public double averageOutside;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy