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

boofcv.alg.shapes.polygon.RefinePolygonToGrayLine 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.polygon;

import boofcv.alg.shapes.edge.SnapToLineEdge;
import boofcv.struct.distort.PixelTransform;
import boofcv.struct.image.ImageGray;
import georegression.geometry.UtilLine2D_F64;
import georegression.metric.Intersection2D_F64;
import georegression.struct.line.LineGeneral2D_F64;
import georegression.struct.point.Point2D_F32;
import georegression.struct.point.Point2D_F64;
import georegression.struct.shapes.Polygon2D_F64;

/**
 * 

* Improves the fits of a polygon's which is darker or lighter than the background. Info's edges are * assumed to be perfectly straight lines. The edges are processed individually and fit to a line using weighted * regression. Both black squares with white backgrounds and white shapes with black backgrounds can be found. * The edges are selected to maximize the difference between light and dark regions. *

*

* For example, assume an image axis aligned rectangle has a lower extent of 1,2 and a upper extent of 12,15, is * entirely filled, excluding the upper extent pixels (as is typical). Then the found lower and upper extends of the * found polygon will also be 1,2 and 12,15. *

* *

* If a line lies entirely along the image border it is not modified. If part of it lies along the image then only * points not near the border are used to optimize its location. *

* *

For input polygons which are in undistorted coordinates by with a distorted image call {@link #getSnapToEdge()} * and invoke {@link SnapToLineEdge#setTransform(PixelTransform)}.

* * @author Peter Abeles */ public class RefinePolygonToGrayLine> implements RefinePolygonToGray { // How far away from a corner will it sample the line private double cornerOffset = 2.0; // maximum number of iterations private int maxIterations = 10; // convergence tolerance in pixels private double convergeTolPixels = 0.01; // The maximum number of pixels a corner can move in a single iteration // designed to prevent divergence private double maxCornerChangePixel=2; private SnapToLineEdge snapToEdge; //---------- storage for local work space private LineGeneral2D_F64 general[] = new LineGeneral2D_F64[0]; // estimated line for each side private Polygon2D_F64 previous; // adjusted corner points which have been offset from the true corners private Point2D_F64 adjA = new Point2D_F64(); private Point2D_F64 adjB = new Point2D_F64(); // the input image protected T image; private Class imageType; // work space for checking to see if the line estimate diverged private Point2D_F64 tempA = new Point2D_F64(); private Point2D_F64 tempB = new Point2D_F64(); private LineGeneral2D_F64 before = new LineGeneral2D_F64(); /** * Constructor which provides full access to all parameters. See code documents * value a description of these variables. * */ public RefinePolygonToGrayLine(double cornerOffset, int lineSamples, int sampleRadius, int maxIterations, double convergeTolPixels, double maxCornerChangePixel, Class imageType ) { this.cornerOffset = cornerOffset; this.maxIterations = maxIterations; this.convergeTolPixels = convergeTolPixels; this.maxCornerChangePixel = maxCornerChangePixel; this.snapToEdge = new SnapToLineEdge<>(lineSamples, sampleRadius, imageType); this.imageType = imageType; previous = new Polygon2D_F64(1); } /** * Simplified constructor which uses reasonable default values for most variables * @param numSides Number of sides on the polygon * @param imageType Type of input image it processes */ public RefinePolygonToGrayLine(int numSides , Class imageType) { previous = new Polygon2D_F64(numSides); this.imageType = imageType; this.snapToEdge = new SnapToLineEdge<>(20, 1, imageType); } /** * Sets the image which is going to be processed. If a transform is to be used * {@link SnapToLineEdge#setTransform} should be called before this. */ @Override public void setImage(T image) { this.image = image; this.snapToEdge.setImage(image); } @Override public void setLensDistortion(int width, int height, PixelTransform distToUndist, PixelTransform undistToDist) { this.snapToEdge.setTransform(undistToDist); } @Override public void clearLensDistortion() { this.snapToEdge.setTransform(null); } /** * Refines the fit a polygon by snapping it to the edges. * * @param input (input) Initial estimate for the polygon. CW or CCW ordering doesn't matter. * @param output (output) the fitted polygon */ @Override public boolean refine(Polygon2D_F64 input, Polygon2D_F64 output) { if( input.size() != output.size()) throw new IllegalArgumentException("Input and output sides do not match. "+input.size()+" "+output.size()); // sanity check input. If it's too small this algorithm won't work if( checkShapeTooSmall(input) ) return false; // see if this work space needs to be resized if( general.length < input.size() ) { general = new LineGeneral2D_F64[input.size() ]; for (int i = 0; i < general.length; i++) { general[i] = new LineGeneral2D_F64(); } } // estimate line equations return optimize(input,output); } /** * Looks at the distance between each vertex. If that distance is so small the edge can't be measured the * return true. * @param input polygon * @return true if too small or false if not */ private boolean checkShapeTooSmall(Polygon2D_F64 input) { // must be longer than the border plus some small fudge factor double minLength = cornerOffset*2 + 2; for (int i = 0; i < input.size(); i++) { int j = (i+1)%input.size(); Point2D_F64 a = input.get(i); Point2D_F64 b = input.get(j); if( a.distance2(b) < minLength*minLength ) return true; } return false; } /** * Refines the initial line estimates using EM. The number of iterations is fixed. */ protected boolean optimize(Polygon2D_F64 seed , Polygon2D_F64 current ) { previous.set(seed); // pixels squares is faster to compute double convergeTol = convergeTolPixels*convergeTolPixels; // initialize the lines since they are used to check for corner divergence for (int i = 0; i < seed.size(); i++) { int j = (i + 1) % seed.size(); Point2D_F64 a = seed.get(i); Point2D_F64 b = seed.get(j); UtilLine2D_F64.convert(a,b,general[i]); } boolean changed = false; for (int iteration = 0; iteration < maxIterations; iteration++) { // snap each line to the edge independently. Lines will be in local coordinates for (int i = 0; i < previous.size(); i++) { int j = (i + 1) % previous.size(); Point2D_F64 a = previous.get(i); Point2D_F64 b = previous.get(j); before.set(general[i]); boolean failed = false; if( !optimize(a,b,general[i]) ) { failed = true; } else { int k = (i+previous.size()-1) %previous.size(); // see if the corner has diverged if( Intersection2D_F64.intersection(general[k], general[i],tempA) != null && Intersection2D_F64.intersection(general[i], general[j],tempB) != null ) { if( tempA.distance(a) > maxCornerChangePixel || tempB.distance(b) > maxCornerChangePixel ) { failed = true; } } else { failed = true; } } // The line fit failed. Probably because its along the image border. Revert it if( failed ) { general[i].set(before); } else { changed = true; } } // Find the corners of the quadrilateral from the lines if( !UtilShapePolygon.convert(general,current) ) return false; // see if it has converged boolean converged = true; for (int i = 0; i < current.size(); i++) { if( current.get(i).distance2(previous.get(i)) > convergeTol ) { converged = false; break; } } if( converged ) { // System.out.println("Converged early at "+iteration); break; } else { previous.set(current); } } return changed; } /** * Fits a line defined by the two points. When fitting the line the weight of the edge is used to determine * how influential the point is * @param a Corner point in image coordinates. * @param b Corner point in image coordinates. * @param found (output) Line in image coordinates * @return true if successful or false if it failed */ protected boolean optimize( Point2D_F64 a , Point2D_F64 b , LineGeneral2D_F64 found ) { computeAdjustedEndPoints(a, b); return snapToEdge.refine(adjA, adjB, found); } /** * Used to specify a transform that is applied to pixel coordinates to bring them back into original input * image coordinates. For example if the input image has lens distortion but the edge were found * in undistorted coordinates this code needs to know how to go from undistorted back into distorted * image coordinates in order to read the pixel's value. * * @param undistToDist Pixel transformation from undistorted pixels into the actual distorted input image.. */ public void setTransform( PixelTransform undistToDist ) { snapToEdge.setTransform(undistToDist); } private void computeAdjustedEndPoints(Point2D_F64 a, Point2D_F64 b) { double slopeX = (b.x - a.x); double slopeY = (b.y - a.y); double r = Math.sqrt(slopeX*slopeX + slopeY*slopeY); // vector of unit length pointing in direction of the slope double unitX = slopeX/r; double unitY = slopeY/r; // offset from corner because the gradient because unstable around there adjA.x = a.x + unitX*cornerOffset; adjA.y = a.y + unitY*cornerOffset; adjB.x = b.x - unitX*cornerOffset; adjB.y = b.y - unitY*cornerOffset; } public SnapToLineEdge getSnapToEdge() { return snapToEdge; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy