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

boofcv.alg.shapes.edge.SnapToLineEdge 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-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.edge;

import boofcv.struct.image.ImageGray;
import georegression.fitting.line.FitLine_F64;
import georegression.geometry.UtilLine2D_F64;
import georegression.struct.line.LineGeneral2D_F64;
import georegression.struct.line.LinePolar2D_F64;
import georegression.struct.point.Point2D_F64;
import org.ddogleg.struct.FastQueue;
import org.ddogleg.struct.GrowQueue_F64;

/**
 * 

* Snaps a line to an edge of an object. The refined line attempts to maximize the absolute value of the * difference between the left and right sides of the line. *

* *

* The algorithm works by sampling along the provided line segment. For each point along the line it also * samples points tangential to it in the left and right direction. When a point is sampled it is actually * the line integral between two points which are one pixel apart. The weight is found as the absolute value of * difference the between two adjacent line integrals along the tangent. *

*

* Internally it will compute the solution in a local coordinate system to reduce numerical errors. *

* *

* DISTORTED INPUT IMAGE: If the distortion is known it is possible to sample along a straight line in distorted * image space. This can be accomplished through the use of {@link #setTransform} where the provided transform * goes from undistorted pixel coordinates into the distorted input image. *

* * @author Peter Abeles */ // TODO specify weight function. close to A, close to B, even public class SnapToLineEdge>extends BaseIntegralEdge { // maximum number of times it will sample along the line protected int lineSamples; // Determines the number of points sampled radially outwards from the line // Total intensity values sampled at each point along the line is radius*2+2, // and points added to line fitting is radius*2+1. protected int radialSamples; // storage for computed polar line private LinePolar2D_F64 polar = new LinePolar2D_F64(); protected GrowQueue_F64 weights = new GrowQueue_F64();// storage for weights in line fitting // storage for where the points that are sampled along the line protected FastQueue samplePts = new FastQueue<>(Point2D_F64::new); // storage for the line's center. used to reduce numerical problems. protected Point2D_F64 center = new Point2D_F64(); protected double localScale; /** * Configures the algorithm. * * @param lineSamples Number of times it will sample along the line's axis. Try 19 * @param tangentialSamples Radius along the tangent of what it will sample. Must be ≥ 1. Try 2. * @param imageType Type of image it's going to process */ public SnapToLineEdge(int lineSamples, int tangentialSamples, Class imageType) { super(imageType); if( tangentialSamples < 1 ) throw new IllegalArgumentException("Tangential samples must be >= 1 or else it won't work"); this.lineSamples = lineSamples; this.radialSamples = tangentialSamples; } /** * 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. Multiple calls might be required to get a perfect fit. * * @param a Start of line * @param b End of line.. * @param found (output) Fitted line to the edge * @return true if successful or false if it failed */ public boolean refine(Point2D_F64 a, Point2D_F64 b, LineGeneral2D_F64 found) { // determine the local coordinate system center.x = (a.x + b.x)/2.0; center.y = (a.y + b.y)/2.0; localScale = a.distance(center); // define the line which points are going to be sampled along double slopeX = (b.x - a.x); double slopeY = (b.y - a.y); double r = Math.sqrt(slopeX*slopeX + slopeY*slopeY); // tangent of unit length that radial sample samples are going to be along // Two choices for tangent here. Select the one which points to the "right" of the line, // which is inside of the edge double tanX = slopeY/r; double tanY = -slopeX/r; // set up inputs into line fitting computePointsAndWeights(slopeX, slopeY, a.x, a.y, tanX, tanY); if( samplePts.size() >= 4 ) { // fit line and convert into generalized format if( null == FitLine_F64.polar(samplePts.toList(), weights.data, polar) ) { throw new RuntimeException("All weights were zero, bug some place"); } UtilLine2D_F64.convert(polar, found); // Convert line from local to global coordinates localToGlobal(found); return true; } else { return false; } } /** * Computes the location of points along the line and their weights */ protected void computePointsAndWeights(double slopeX, double slopeY, double x0, double y0, double tanX, double tanY) { samplePts.reset(); weights.reset(); int numSamples = radialSamples*2+2; int numPts = numSamples-1; double widthX = numSamples*tanX; double widthY = numSamples*tanY; for (int i = 0; i < lineSamples; i++ ) { // find point on line and shift it over to the first sample point double frac = i/(double)(lineSamples -1); double x = x0 + slopeX*frac-widthX/2.0; double y = y0 + slopeY*frac-widthY/2.0; // Unless all the sample points are inside the image, ignore this point if(!integral.isInside(x, y) || !integral.isInside(x + widthX,y + widthY)) continue; double sample0 = integral.compute(x, y, x + tanX, y + tanY); x += tanX; y += tanY; for (int j = 0; j < numPts; j++) { // still need to check the next point due to round off error. very rare condition if( j==numPts-1&&!integral.isInside(x + tanX, y + tanY)) break; double sample1 = integral.compute(x, y, x + tanX, y + tanY); double w = sample0 - sample1; if( w < 0 ) w = -w; if( w > 0 ) { weights.add(w); samplePts.grow().set((x - center.x) / localScale, (y - center.y) / localScale); } x += tanX; y += tanY; sample0 = sample1; } } } /** * Converts the line from local to global image coordinates */ protected void localToGlobal( LineGeneral2D_F64 line ) { line.C = localScale*line.C - center.x*line.A - center.y*line.B; } public int getLineSamples() { return lineSamples; } public void setLineSamples(int lineSamples) { this.lineSamples = lineSamples; } public int getRadialSamples() { return radialSamples; } public void setRadialSamples(int radialSamples) { this.radialSamples = radialSamples; } public Class getImageType() { return imageType; } public void setImageType(Class imageType) { this.imageType = imageType; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy