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

boofcv.alg.feature.detect.interest.FeatureLaplacePyramid 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.interest;

import boofcv.abst.feature.detect.interest.InterestPointScaleSpacePyramid;
import boofcv.abst.filter.ImageFunctionSparse;
import boofcv.abst.filter.derivative.AnyImageDerivative;
import boofcv.struct.QueueCorner;
import boofcv.struct.feature.ScalePoint;
import boofcv.struct.image.GrayF32;
import boofcv.struct.image.ImageGray;
import boofcv.struct.pyramid.PyramidFloat;
import georegression.struct.point.Point2D_I16;

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

import static boofcv.alg.feature.detect.interest.FastHessianFeatureDetector.polyPeak;

/**
 * 

* Feature detector across image pyramids that uses the Laplacian to determine strength in scale-space. *

*

* COMMENT ON SCALEPOWER: To normalize feature intensity across scales each feature intensity is multiplied by the scale to the power of 'scalePower'. * See [1,2] for how to compute 'scalePower'. Inside of the image pyramid sub-sampling of the image causes the image * gradient to be a factor of 'scale' larger than it would be without sub-sampling. In some situations this can negate * the need to adjust feature intensity further. *

*

* [1] Krystian Mikolajczyk and Cordelia Schmid, "Indexing based on scale invariant interest points" ICCV 2001. Proceedings.
* [2] Lindeberg, T., "Feature detection with automatic scale selection." IJCV 30(2) (1998) 79 – 116 *

* * @author Peter Abeles * @see boofcv.factory.feature.detect.interest.FactoryInterestPoint */ public class FeatureLaplacePyramid, D extends ImageGray> implements InterestPointScaleSpacePyramid { // used to compute feature intensity across scale space private ImageFunctionSparse sparseLaplace; // generalized feature detector. Used to find candidate features in each scale's image private GeneralFeatureDetector detector; private float baseThreshold; // location of recently computed features in layers protected int spaceIndex = 0; protected List maximums = new ArrayList<>(); // List of found feature points protected List foundPoints = new ArrayList<>(); protected AnyImageDerivative computeDerivative; // how much the feature intensity is scaled in each level // varies depending on feature type, used to adjust detection threshold protected double scalePower; /** * Create a feature detector. * * @param detector Point feature detector which is used to find candidates in each scale level * @param sparseLaplace Used to compute the Laplacian at each candidates * @param computeDerivative Used to compute image derivatives * @param scalePower Used to normalize features intensity across scale space. For many features this value should be one. */ public FeatureLaplacePyramid(GeneralFeatureDetector detector, ImageFunctionSparse sparseLaplace, AnyImageDerivative computeDerivative, double scalePower) { this.detector = detector; this.baseThreshold = detector.getThreshold(); this.computeDerivative = computeDerivative; this.sparseLaplace = sparseLaplace; this.scalePower = scalePower; } /** * Searches for features inside the provided scale space * * @param ss Scale space of an image */ @Override public void detect(PyramidFloat ss) { spaceIndex = 0; foundPoints.clear(); // compute feature intensity in each level for (int i = 1; i < ss.getNumLayers()-1; i++) { // detect features in 2D space. Don't need to compute features at the tail ends of scale-space // if (i > 0 && i < ss.getNumLayers() - 1) // detectCandidateFeatures(ss.getLayer(i), ss.getSigma(i)); spaceIndex = i; detectCandidateFeatures(ss.getLayer(i), ss.getSigma(i)); // find maximum in 3xNx3 (local image and scale space) region findLocalScaleSpaceMax(ss, i); // spaceIndex++; // if (spaceIndex >= 3) // spaceIndex = 0; // // // find maximum in 3x3x3 (local image and scale space) region // if (i >= 2) { // detectCandidateFeatures(ss.getLayer(i-i), ss.getSigma(i-1)); // findLocalScaleSpaceMax(ss, i - 1); // } } } /** * Use the feature detector to find candidate features in each level. Only compute the needed image derivatives. */ private void detectCandidateFeatures(T image, double sigma ) { // adjust corner intensity threshold based upon the current scale factor float scaleThreshold = (float) (baseThreshold / Math.pow(sigma, scalePower)); detector.setThreshold(scaleThreshold); computeDerivative.setInput(image); D derivX = null, derivY = null; D derivXX = null, derivYY = null, derivXY = null; if (detector.getRequiresGradient()) { derivX = computeDerivative.getDerivative(true); derivY = computeDerivative.getDerivative(false); } if (detector.getRequiresHessian()) { derivXX = computeDerivative.getDerivative(true, true); derivYY = computeDerivative.getDerivative(false, false); derivXY = computeDerivative.getDerivative(true, false); } detector.process(image, derivX, derivY, derivXX, derivYY, derivXY); List m = maximums; m.clear(); if( detector.isDetectMaximums() ) { QueueCorner q = detector.getMaximums(); for (int i = 0; i < q.size; i++) { m.add(q.get(i).copy()); } } if( detector.isDetectMinimums() ) { QueueCorner q = detector.getMinimums(); for (int i = 0; i < q.size; i++) { m.add(q.get(i).copy()); } } } /** * See if each feature is a maximum in its local scale-space. */ protected void findLocalScaleSpaceMax(PyramidFloat ss, int layerID) { List candidates = maximums; float scale0 = (float) ss.scale[layerID - 1]; float scale1 = (float) ss.scale[layerID]; float scale2 = (float) ss.scale[layerID + 1]; float sigma0 = (float) ss.getSigma(layerID - 1); float sigma1 = (float) ss.getSigma(layerID); float sigma2 = (float) ss.getSigma(layerID + 1); // For laplacian its t^(2*gamma) where gamma = 3/4 float ss0 = (float) (Math.pow(sigma0, 2.0 * 0.75)/scale0);// Is this divide by scale correct? float ss1 = (float) (Math.pow(sigma1, 2.0 * 0.75)/scale1); float ss2 = (float) (Math.pow(sigma2, 2.0 * 0.75)/scale2); for (Point2D_I16 c : candidates) { GrayF32 intensity = detector.getIntensity(); float target = intensity.unsafe_get(c.x,c.y); float fx,fy; { float x0 = intensity.unsafe_get(c.x - 1, c.y); float x2 = intensity.unsafe_get(c.x + 1, c.y); float y0 = intensity.unsafe_get(c.x, c.y - 1); float y2 = intensity.unsafe_get(c.x, c.y + 1); fx = c.x + polyPeak(x0, target, x2); fy = c.y + polyPeak(y0, target, y2); } // fx=c.x;fy=c.y; sparseLaplace.setImage(ss.getLayer(layerID)); float val = ss1 * (float) sparseLaplace.compute(c.x,c.y); // search for local maximum or local minimum float adj = Math.signum(val); val *= adj; // find pixel location in each image's local coordinate int x0 = (int) (fx * scale1 / scale0 + 0.5); int y0 = (int) (fy * scale1 / scale0 + 0.5); int x2 = (int) (fx * scale1 / scale2 + 0.5); int y2 = (int) (fy * scale1 / scale2 + 0.5); if (checkMax(ss.getLayer(layerID - 1), adj*ss0,val, x0, y0) && checkMax(ss.getLayer(layerID + 1), adj*ss2,val, x2, y2)) { sparseLaplace.setImage(ss.getLayer(layerID-1)); float s0 = ss0 * (float) sparseLaplace.compute(x0,y0)*adj; sparseLaplace.setImage(ss.getLayer(layerID+1)); float s2 = ss2 * (float) sparseLaplace.compute(x2,y2)*adj; double adjSigma; double sigmaInterp = polyPeak(s0, val, s2); // scaled from -1 to 1 if( sigmaInterp < 0 ) { adjSigma = sigma0*(-sigmaInterp) + (1+sigmaInterp)*sigma1; } else { adjSigma = sigma2*sigmaInterp + (1-sigmaInterp)*sigma1; } // put features into the scale of the upper image foundPoints.add(new ScalePoint(fx * scale1, fy * scale1, adjSigma)); } } } /** * See if the best score is better than the local adjusted scores at this scale */ private boolean checkMax(T image, double adj, double bestScore, int c_x, int c_y) { sparseLaplace.setImage(image); boolean isMax = true; beginLoop: for (int i = c_y - 1; i <= c_y + 1; i++) { for (int j = c_x - 1; j <= c_x + 1; j++) { double value = adj*sparseLaplace.compute(j, i); if (value >= bestScore) { isMax = false; break beginLoop; } } } return isMax; } @Override public List getInterestPoints() { return foundPoints; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy