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

boofcv.alg.feature.detect.line.HoughTransformBinary 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.feature.detect.line;


import boofcv.abst.feature.detect.extract.NonMaxSuppression;
import boofcv.alg.misc.ImageMiscOps;
import boofcv.struct.ConfigLength;
import boofcv.struct.QueueCorner;
import boofcv.struct.image.GrayF32;
import boofcv.struct.image.GrayU8;
import georegression.struct.line.LineParametric2D_F32;
import georegression.struct.point.Point2D_I16;
import org.ddogleg.struct.FastQueue;
import org.ddogleg.struct.GrowQueue_F32;

import java.util.ArrayList;
import java.util.List;

/**
 * 

* Hough transform which uses a polar line representation, distance from origin and angle (0 to 180 degrees). * Standard implementation of a hough transform. 1) Gradient intensity image is used to find edge pixels. * 2) All possible lines passing through that point are found. 3) Line parameters are summed up in the line image, * in which each pixel represents a coordinate in parameter space. * 3) Local maximums are found. *

*

By the nature of this algorithms, lines are forced to be discretized into parameter space. The range * can vary from +- the maximum range inside the image and the angle from 0 to PI radians. How * finely discretized an image is effects line detection accuracy. If too fine lines might not be detected * or it will be too noisy. *

*

* In the line image, the transform from line parameter coordinate to pixel coordinate is as follow:
* x = r*cos(theta) + centerX
* y = r*sin(theta) + centerY
*

* *

* USAGE NOTE: Duplicate/very similar lines are possible due to angles being cyclical. What happens is that if * a line's orientation lies along a boundary point its angles will be split up between top and bottom * of the transform. When lines are extracted using non-maximum it will detects peaks at the top * and bottom. *

* * @author Peter Abeles */ public class HoughTransformBinary { // extracts line from the transform NonMaxSuppression extractor; // stores returned lines FastQueue linesAll = new FastQueue<>(10, LineParametric2D_F32::new); // Lines after similar ones have been merged together List linesMerged = new ArrayList<>(); // contains a set of counts for detected lines in each pixel // floating point image used because that's what FeatureExtractor's take as input GrayF32 transform = new GrayF32(1,1); // found lines in transform space QueueCorner foundPeaks = new QueueCorner(10); // line intensities for later pruning GrowQueue_F32 foundIntensity = new GrowQueue_F32(10); HoughTransformParameters parameters; // post processing pruning ImageLinePruneMerge post = new ImageLinePruneMerge(); // tuning parameters for merging double mergeAngle = Math.PI*0.05; double mergeDistance = 10; int maxLines = 0; // threshold for number of counts. relative is relative to total area of transform. fixed is number of counts ConfigLength thresholdCounts = ConfigLength.relative(0.001,1); /** * Specifies parameters of transform. The minimum number of points specified in the extractor * is an important tuning parameter. * * @param extractor Extracts local maxima from transform space. */ public HoughTransformBinary(NonMaxSuppression extractor , HoughTransformParameters parameters) { this.extractor = extractor; this.parameters = parameters; } /** * Computes the Hough transform of the image. * * @param binary Binary image that indicates which pixels lie on edges. */ public void transform( GrayU8 binary ) { parameters.initialize(binary.width,binary.height,transform); ImageMiscOps.fill(transform, 0); computeParameters(binary); extractLines(); if( maxLines <= 0 ) { linesMerged.clear(); linesMerged.addAll(linesAll.toList()); } else { mergeLines(binary.width,binary.height); } } void computeParameters(GrayU8 binary) { for( int y = 0; y < binary.height; y++ ) { int start = binary.startIndex + y*binary.stride; int stop = start + binary.width; for( int index = start; index < stop; index++ ) { if( binary.data[index] != 0 ) { parameters.parameterize(index-start,y,transform); } } } } /** * Searches for local maximals and converts into lines. * * @return Found lines in the image. */ protected void extractLines() { linesAll.reset(); foundPeaks.reset(); foundIntensity.reset(); extractor.setThresholdMaximum((float) thresholdCounts.compute(transform.width*transform.height)); extractor.process(transform, null,null,null, foundPeaks); for(int i = 0; i < foundPeaks.size(); i++ ) { Point2D_I16 p = foundPeaks.get(i); if( !parameters.isTransformValid(p.x,p.y)) continue; parameters.transformToLine(p.x,p.y, linesAll.grow()); foundIntensity.push( transform.get(p.x,p.y)); } } protected void mergeLines( int width , int height ) { post.reset(); for( int i = 0; i < linesAll.size(); i++ ) { post.add(linesAll.get(i),foundIntensity.get(i)); } // NOTE: angular accuracy is a function of range from sub image center. This pruning // function uses a constant value for range accuracy. A custom algorithm should really // be used here. post.pruneSimilar((float) mergeAngle, (float)mergeDistance, width, height); post.pruneNBest(maxLines); post.createList(linesMerged); } /** // * Returns the Hough transform image. * * @return Transform image. */ public GrayF32 getTransform() { return transform; } /** * Returns the intensity/edge count for each returned line. Useful when doing * post processing pruning. * * @return Array containing line intensities. */ public float[] getFoundIntensity() { return foundIntensity.data; } public FastQueue getLinesAll() { return linesAll; } public List getLinesMerged() { return linesMerged; } public double getMergeAngle() { return mergeAngle; } public void setMergeAngle(double mergeAngle) { this.mergeAngle = mergeAngle; } public double getMergeDistance() { return mergeDistance; } public void setMergeDistance(double mergeDistance) { this.mergeDistance = mergeDistance; } public int getMaxLines() { return maxLines; } public void setMaxLines(int maxLines) { this.maxLines = maxLines; } public HoughTransformParameters getParameters() { return parameters; } public void setNumberOfCounts( ConfigLength counts ) { this.thresholdCounts = counts; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy