
com.irurueta.ar.epipolar.refiners.RightEpipoleRefiner Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of irurueta-ar Show documentation
Show all versions of irurueta-ar Show documentation
Augmented Reality and 3D reconstruction library
/*
* Copyright (C) 2017 Alberto Irurueta Carro ([email protected])
*
* 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 com.irurueta.ar.epipolar.refiners;
import com.irurueta.ar.epipolar.FundamentalMatrix;
import com.irurueta.ar.epipolar.InvalidFundamentalMatrixException;
import com.irurueta.geometry.HomogeneousPoint2D;
import com.irurueta.geometry.Line2D;
import com.irurueta.geometry.Point2D;
import com.irurueta.geometry.Transformation2D;
import com.irurueta.geometry.estimators.LockedException;
import com.irurueta.geometry.estimators.NotReadyException;
import com.irurueta.geometry.refiners.PairMatchesAndInliersDataRefiner;
import com.irurueta.geometry.refiners.RefinerException;
import com.irurueta.numerical.robust.InliersData;
import java.util.BitSet;
import java.util.List;
/**
* Base class to refine the epipole of a fundamental matrix formed by an initial
* epipole estimation and an estimated homography.
* Any fundamental matrix can be expressed as F = [e']x*H, where
* e' is the epipole on the right view and H is a non degenerate homography.
* This class refines an initial epipole so that residuals from provided point
* correspondences generating fundamental matrix F are reduced.
* This class is especially useful in cases where geometry of the scene is
* degenerate (e.g. planar scene) and provided point correspondences would
* generate an inaccurate fundamental matrix.
*/
@SuppressWarnings("DuplicatedCode")
public abstract class RightEpipoleRefiner extends
PairMatchesAndInliersDataRefiner {
/**
* Test line to compute epipolar residuals.
*/
protected final Line2D mTestLine = new Line2D();
/**
* Homography relating two views through a given planar scene.
*/
protected Transformation2D mHomography;
/**
* Standard deviation used for Levenberg-Marquardt fitting during
* refinement.
* Returned value gives an indication of how much variance each residual
* has.
* Typically this value is related to the threshold used on each robust
* estimation, since residuals of found inliers are within the range of
* such threshold.
*/
private double mRefinementStandardDeviation;
/**
* Constructor.
*/
protected RightEpipoleRefiner() {
}
/**
* Constructor.
*
* @param initialEpipoleEstimation initial right epipole estimation to be
* set and refined.
* @param keepCovariance true if covariance of estimation must be kept after
* refinement, false otherwise.
* @param inliers set indicating which of the provided matches are inliers.
* @param residuals residuals for matched samples.
* @param numInliers number of inliers on initial estimation.
* @param samples1 1st set of paired samples.
* @param samples2 2nd set of paired samples.
* @param refinementStandardDeviation standard deviation used for
* Levenberg-Marquardt fitting.
* @param homography homography relating samples in two views, which is used
* to generate a fundamental matrix and its corresponding epipolar geometry.
*/
protected RightEpipoleRefiner(
final Point2D initialEpipoleEstimation,
final boolean keepCovariance, final BitSet inliers, final double[] residuals,
final int numInliers, final List samples1, final List samples2,
final double refinementStandardDeviation, final Transformation2D homography) {
super(initialEpipoleEstimation, keepCovariance, inliers, residuals,
numInliers, samples1, samples2);
mHomography = homography;
mRefinementStandardDeviation = refinementStandardDeviation;
}
/**
* Constructor.
*
* @param initialEpipoleEstimation initial right epipole estimation to be
* set and refined.
* @param keepCovariance true if covariance of estimation must be kept after
* refinement, false otherwise.
* @param inliersData inlier data, typically obtained from a robust
* estimator.
* @param samples1 1st set of paired samples.
* @param samples2 2nd set of paired samples.
* @param refinementStandardDeviation standard deviation used for
* Levenberg-Marquardt fitting.
* @param homography homography relating samples in two views, which is used
* to generate a fundamental matrix and its corresponding epipolar geometry.
*/
protected RightEpipoleRefiner(
final Point2D initialEpipoleEstimation,
final boolean keepCovariance, final InliersData inliersData,
final List samples1, final List samples2,
final double refinementStandardDeviation, final Transformation2D homography) {
super(initialEpipoleEstimation, keepCovariance, inliersData, samples1,
samples2);
mHomography = homography;
mRefinementStandardDeviation = refinementStandardDeviation;
}
/**
* Gets standard deviation used for Levenberg-Marquardt fitting during
* refinement.
* Returned value gives an indication of how much variance each residual
* has.
* Typically this value is related to the threshold used on each robust
* estimation, since residuals of found inliers are within the range of such
* threshold.
*
* @return standard deviation used for refinement.
*/
public double getRefinementStandardDeviation() {
return mRefinementStandardDeviation;
}
/**
* Sets standard deviation used for Levenberg-Marquardt fitting during
* refinement.
* Returned value gives an indication of how much variance each residual
* has.
* Typically this value is related to the threshold used on each robust
* estimation, since residuals of found inliers are within the range of such
* threshold.
*
* @param refinementStandardDeviation standard deviation used for
* refinement.
* @throws LockedException if estimator is locked.
*/
public void setRefinementStandardDeviation(
final double refinementStandardDeviation) throws LockedException {
if (isLocked()) {
throw new LockedException();
}
mRefinementStandardDeviation = refinementStandardDeviation;
}
/**
* Gets homography relating samples in two views, which is used to generate
* a fundamental matrix and its corresponding epipolar geometry.
*
* @return homography relating samples in two views.
*/
public Transformation2D getHomography() {
return mHomography;
}
/**
* Sets homography relating samples in two views, which is used to generate
* a fundamental matrix and its corresponding epipolar geometry.
*
* @param homography homography relating samples in two views.
* @throws LockedException if estimator is locked.
*/
public void setHomography(final Transformation2D homography)
throws LockedException {
if (isLocked()) {
throw new LockedException();
}
mHomography = homography;
}
/**
* Indicates whether this refiner is ready to start refinement computation.
*
* @return true if refiner is ready, false otherwise.
*/
@Override
public boolean isReady() {
return mHomography != null && super.isReady();
}
/**
* Refines provided initial right epipole estimation.
*
* @return refined estimation.
* @throws NotReadyException if not enough input data has been provided.
* @throws LockedException if estimator is locked because refinement is
* already in progress.
* @throws RefinerException if refinement fails for some reason (e.g. unable
* to converge to a result).
*/
@Override
public Point2D refine() throws NotReadyException, LockedException,
RefinerException {
final HomogeneousPoint2D result = new HomogeneousPoint2D();
refine(result);
return result;
}
/**
* Computes a fundamental matrix from a 2D homography and provided epipole
* on right view.
*
* @param homography a 2D homography. Must be invertible.
* @param rightEpipole epipole on right view.
* @param result instance where computed fundamental matrix will be stored.
* @throws InvalidFundamentalMatrixException if provided homography is not
* invertible, which would generate a degenerate fundamental matrix.
*/
public static void computeFundamentalMatrix(
final Transformation2D homography,
final Point2D rightEpipole,
final FundamentalMatrix result)
throws InvalidFundamentalMatrixException {
result.setFromHomography(homography, rightEpipole);
}
/**
* Computes the residual between a fundamental matrix and a pair of matched
* points.
*
* @param fundamentalMatrix a fundamental matrix.
* @param leftPoint left 2D point.
* @param rightPoint right 2D point.
* @return residual (distance of point to epipolar line).
*/
protected double residual(
final FundamentalMatrix fundamentalMatrix,
final Point2D leftPoint,
final Point2D rightPoint) {
try {
leftPoint.normalize();
rightPoint.normalize();
fundamentalMatrix.normalize();
fundamentalMatrix.leftEpipolarLine(rightPoint, mTestLine);
final double leftDistance = Math.abs(mTestLine.signedDistance(
leftPoint));
fundamentalMatrix.rightEpipolarLine(leftPoint, mTestLine);
final double rightDistance = Math.abs(mTestLine.signedDistance(
rightPoint));
// return average distance as an error residual
return 0.5 * (leftDistance + rightDistance);
} catch (final NotReadyException e) {
return Double.MAX_VALUE;
}
}
/**
* Computes total residual among all provided inlier samples.
*
* @param fundamentalMatrix a fundamental matrix.
* @return total residual.
*/
protected double totalResidual(
final FundamentalMatrix fundamentalMatrix) {
double result = 0.0;
final int nSamples = mInliers.length();
Point2D leftPoint;
Point2D rightPoint;
for (int i = 0; i < nSamples; i++) {
if (mInliers.get(i)) {
// sample is inlier
leftPoint = mSamples1.get(i);
rightPoint = mSamples2.get(i);
leftPoint.normalize();
rightPoint.normalize();
result += residual(fundamentalMatrix, leftPoint, rightPoint);
}
}
return result;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy