boofcv.alg.feature.detect.template.TemplateMatching 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.template;
import boofcv.abst.feature.detect.extract.ConfigExtract;
import boofcv.abst.feature.detect.extract.NonMaxSuppression;
import boofcv.factory.feature.detect.extract.FactoryFeatureExtractor;
import boofcv.struct.QueueCorner;
import boofcv.struct.feature.Match;
import boofcv.struct.image.GrayF32;
import boofcv.struct.image.ImageBase;
import georegression.struct.point.Point2D_I16;
import org.ddogleg.sorting.QuickSelect;
import org.ddogleg.struct.FastQueue;
/**
* Runs a template matching algorithm across the image. Local peaks are found in the resulting
* intensity image and the best solutions returned.
*
* @author Peter Abeles
*/
public class TemplateMatching> {
// computes an intensity image identifying matches
private TemplateMatchingIntensity match;
// searches for local matches in intensity image
private NonMaxSuppression extractor;
// Reference to the template being searched for
private T template;
// Mask for template to determine influence of each pixel
private T mask;
// Maximum number of matches that can be returned
private int maxMatches;
// storage for found points
private QueueCorner candidates = new QueueCorner(10);
// working space for sorting the results
private float scores[] = new float[10];
private int indexes[] = new int[10];
// storage for final points
private FastQueue results = new FastQueue<>(10, Match::new);
// shape of input image
int imageWidth,imageHeight;
/**
* Specifies internal algorithm
*
* @param match Provide template matching intensity algorithm
*/
public TemplateMatching(TemplateMatchingIntensity match) {
this.match = match;
ConfigExtract config;
if( match.isMaximize() ) {
config = new ConfigExtract(2, -Float.MAX_VALUE, 0, true);
} else {
config = new ConfigExtract(2, -Float.MAX_VALUE, 0, true, true, false);
}
extractor = FactoryFeatureExtractor.nonmax(config);
}
/**
* Adjust how close to objects can be found to each other
*
* @param radius Distance in pixels. Try using the template's radius or 2
*/
public void setMinimumSeparation(int radius) {
extractor.setSearchRadius(radius);
}
/**
* Specifies the template to search for and the maximum number of matches to return.
*
* @param template Template being searched for
* @param mask Optional mask. Same size as template. 0 = pixel is transparent, values larger than zero
* determine how influential the pixel is. Can be null.
* @param maxMatches The maximum number of matches it will return
*/
public void setTemplate(T template, T mask , int maxMatches) {
this.template = template;
this.mask = mask;
this.maxMatches = maxMatches;
}
/**
* Specifies the input image which the template is to be found inside.
*
* @param image Image being processed
*/
public void setImage(T image ) {
match.setInputImage(image);
this.imageWidth = image.width;
this.imageHeight = image.height;
}
/**
* Performs template matching.
*/
public void process() {
// compute match intensities
if( mask == null )
match.process(template);
else
match.process(template,mask);
GrayF32 intensity = match.getIntensity();
int offsetX = 0;
int offsetY = 0;
// adjust intensity image size depending on if there is a border or not
if (!match.isBorderProcessed()) {
int x0 = match.getBorderX0();
int x1 = imageWidth - match.getBorderX1();
int y0 = match.getBorderY0();
int y1 = imageHeight - match.getBorderY1();
intensity = intensity.subimage(x0, y0, x1+1, y1+1, null);
} else {
offsetX = match.getBorderX0();
offsetY = match.getBorderY0();
}
// find local peaks in intensity image
candidates.reset();
if( match.isMaximize() )
extractor.process(intensity, null,null,null, candidates);
else
extractor.process(intensity, null,null,candidates, null);
// select the best matches
if (scores.length < candidates.size) {
scores = new float[candidates.size];
indexes = new int[candidates.size];
}
float sgn = match.isMaximize() ? -1.0f : 1.0f;
for (int i = 0; i < candidates.size; i++) {
Point2D_I16 p = candidates.get(i);
scores[i] = sgn*intensity.get(p.x, p.y);
}
int N = Math.min(maxMatches, candidates.size);
QuickSelect.selectIndex(scores, N, candidates.size, indexes);
// save the results
results.reset();
for (int i = 0; i < N; i++) {
Point2D_I16 p = candidates.get(indexes[i]);
Match m = results.grow();
m.score = -scores[indexes[i]];
m.set(p.x - offsetX, p.y - offsetY);
}
}
/**
* Returns all the found matches. The location is the location of the top left corner
* of the template. Score is the first score with higher number being better
*
* @return List of found templates
*/
public FastQueue getResults() {
return results;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy