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

boofcv.alg.feature.dense.DescribeDenseSiftAlg 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.dense;

import boofcv.alg.InputSanityCheck;
import boofcv.alg.feature.describe.DescribePointSift;
import boofcv.alg.feature.describe.DescribeSiftCommon;
import boofcv.core.image.FactoryGImageGray;
import boofcv.core.image.GImageGray;
import boofcv.struct.feature.TupleDesc_F64;
import boofcv.struct.image.GrayF32;
import boofcv.struct.image.GrayF64;
import boofcv.struct.image.ImageGray;
import georegression.metric.UtilAngle;
import georegression.struct.point.Point2D_I32;
import org.ddogleg.struct.FastQueue;

/**
 * 

Computes {@link DescribePointSift SIFT} features in a regular grid across an entire image at a single * scale and orientation. This is more computationally efficient than the more generic {@link DescribePointSift} * algorithm because it makes strong assumptions. If given the same center point, an orientation of 0, and * sigmaToPixels is 1, they should produce the same descriptor.

* *

Sampling is done in regular increments in a grid pattern. The example sampling points are computed such that * entire area sampled starts and ends at the most extreme possible pixels. This most likely will require that * the sampling period be adjusted. Multiple descriptors can overlap the same area, so pixel orientation and * magnitude is just computed once and saved.

* * @author Peter Abeles */ public class DescribeDenseSiftAlg> extends DescribeSiftCommon { // sampling period along the image's rows an columns double periodRows; double periodColumns; // wrapper around gradient images so that multiple types are supported GImageGray imageDerivX,imageDerivY; // storage for descriptors FastQueue descriptors; // storage for precomputed angle GrayF64 savedAngle = new GrayF64(1,1); GrayF32 savedMagnitude = new GrayF32(1,1); // saved location of where in the image it sampled FastQueue sampleLocations = new FastQueue<>(Point2D_I32::new); /** * Specifies SIFT descriptor structure and sampling frequency. * @param widthSubregion Width of sub-region in samples. Try 4 * @param widthGrid Width of grid in subregions. Try 4. * @param numHistogramBins Number of bins in histogram. Try 8 * @param weightingSigmaFraction Sigma for Gaussian weighting function is set to this value * region width. Try 0.5 * @param maxDescriptorElementValue Helps with non-affine changes in lighting. See paper. Try 0.2 * @param periodColumns Number of pixels between samples along x-axis * @param periodRows Number of pixels between samples along y-axis * @param derivType Type of input derivative image */ public DescribeDenseSiftAlg(int widthSubregion, int widthGrid, int numHistogramBins, double weightingSigmaFraction , double maxDescriptorElementValue, double periodColumns, double periodRows , Class derivType ) { super(widthSubregion,widthGrid,numHistogramBins,weightingSigmaFraction,maxDescriptorElementValue); this.periodRows = periodRows; this.periodColumns = periodColumns; final int DOF = getDescriptorLength(); imageDerivX = FactoryGImageGray.create(derivType); imageDerivY = FactoryGImageGray.create(derivType); descriptors = new FastQueue<>(()->new TupleDesc_F64(DOF)); } /** * Sets the gradient and precomputes pixel orientation and magnitude * * @param derivX image derivative x-axis * @param derivY image derivative y-axis */ public void setImageGradient(D derivX , D derivY ) { InputSanityCheck.checkSameShape(derivX,derivY); if( derivX.stride != derivY.stride || derivX.startIndex != derivY.startIndex ) throw new IllegalArgumentException("stride and start index must be the same"); savedAngle.reshape(derivX.width,derivX.height); savedMagnitude.reshape(derivX.width,derivX.height); imageDerivX.wrap(derivX); imageDerivY.wrap(derivY); precomputeAngles(derivX); } /** * Computes SIFT descriptors across the entire image */ public void process() { int width = widthSubregion*widthGrid; int radius = width/2; int X0 = radius,X1 = savedAngle.width-radius; int Y0 = radius,Y1 = savedAngle.height-radius; int numX = (int)((X1-X0)/periodColumns); int numY = (int)((Y1-Y0)/periodRows); descriptors.reset(); sampleLocations.reset(); for (int i = 0; i < numY; i++) { int y = (Y1-Y0)*i/(numY-1) + Y0; for (int j = 0; j < numX; j++) { int x = (X1-X0)*j/(numX-1) + X0; TupleDesc_F64 desc = descriptors.grow(); computeDescriptor(x,y,desc); sampleLocations.grow().set(x,y); } } } /** * Computes the angle of each pixel and its gradient magnitude */ void precomputeAngles(D image) { int savecIndex = 0; for (int y = 0; y < image.height; y++) { int pixelIndex = y*image.stride + image.startIndex; for (int x = 0; x < image.width; x++, pixelIndex++, savecIndex++ ) { float spacialDX = imageDerivX.getF(pixelIndex); float spacialDY = imageDerivY.getF(pixelIndex); savedAngle.data[savecIndex] = UtilAngle.domain2PI(Math.atan2(spacialDY,spacialDX)); savedMagnitude.data[savecIndex] = (float)Math.sqrt(spacialDX*spacialDX + spacialDY*spacialDY); } } } /** * Computes the descriptor centered at the specified coordinate * @param cx center of region x-axis * @param cy center of region y-axis * @param desc The descriptor */ public void computeDescriptor( int cx , int cy , TupleDesc_F64 desc ) { desc.fill(0); int widthPixels = widthSubregion*widthGrid; int radius = widthPixels/2; for (int i = 0; i < widthPixels; i++) { int angleIndex = (cy-radius+i)*savedAngle.width + (cx-radius); float subY = i/(float)widthSubregion; for (int j = 0; j < widthPixels; j++, angleIndex++ ) { float subX = j/(float)widthSubregion; double angle = savedAngle.data[angleIndex]; float weightGaussian = gaussianWeight[i*widthPixels+j]; float weightGradient = savedMagnitude.data[angleIndex]; // trilinear interpolation intro descriptor trilinearInterpolation(weightGaussian*weightGradient,subX,subY,angle,desc); } } normalizeDescriptor(desc,maxDescriptorElementValue); } public double getPeriodRows() { return periodRows; } public void setPeriodRows(double periodRows) { this.periodRows = periodRows; } public double getPeriodColumns() { return periodColumns; } public void setPeriodColumns(double periodColumns) { this.periodColumns = periodColumns; } public FastQueue getDescriptors() { return descriptors; } /** * Returns where in the image it sampled the features */ public FastQueue getLocations() { return sampleLocations; } public Class getDerivType () { return (Class)(imageDerivX.getImageType()); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy