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

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

/*
 * 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.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.class, true);

	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.getContourFinder().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