![JAR search and dependency download from the Maven repository](/logo.png)
com.irurueta.ar.calibration.estimators.PROMedSImageOfAbsoluteConicRobustEstimator 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) 2015 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.calibration.estimators;
import com.irurueta.ar.calibration.ImageOfAbsoluteConic;
import com.irurueta.geometry.Transformation2D;
import com.irurueta.geometry.estimators.LockedException;
import com.irurueta.geometry.estimators.NotReadyException;
import com.irurueta.numerical.robust.PROMedSRobustEstimator;
import com.irurueta.numerical.robust.PROMedSRobustEstimatorListener;
import com.irurueta.numerical.robust.RobustEstimator;
import com.irurueta.numerical.robust.RobustEstimatorException;
import com.irurueta.numerical.robust.RobustEstimatorMethod;
import java.util.ArrayList;
import java.util.List;
/**
* Finds the best image of absolute conic (IAC) for provided collection of
* homographies (2D transformations) using PROMedS algorithm.
*/
public class PROMedSImageOfAbsoluteConicRobustEstimator extends
ImageOfAbsoluteConicRobustEstimator {
/**
* Default value to be used for stop threshold. Stop threshold can be used
* to keep the algorithm iterating in case that best estimated threshold
* using median of residuals is not small enough. Once a solution is found
* that generates a threshold below this value, the algorithm will stop.
* Threshold is defined by equations h1'*IAC*h2 = 0 and
* h1'*IAC*h1 = h2'*IAC*h2 --< h1'*IAC*h1 - h2'*IAC*h2 = 0, where
* h1 and h2 are the 1st and 2nd columns of an homography (2D
* transformation).
* These equations are derived from the fact that rotation matrices are
* orthonormal.
*/
public static final double DEFAULT_STOP_THRESHOLD = 1e-6;
/**
* Minimum value that can be set as stop threshold.
* Threshold must be strictly greater than 0.0.
*/
public static final double MIN_STOP_THRESHOLD = 0.0;
/**
* Threshold to be used to keep the algorithm iterating in case that best
* estimated threshold using median of residuals is not small enough. Once
* a solution is found that generates a threshold below this value, the
* algorithm will stop.
* The stop threshold can be used to prevent the PROMedS algorithm iterating
* too many times in cases where samples have a very similar accuracy.
* For instance, in cases where proportion of outliers is very small (close
* to 0%), and samples are very accurate (i.e. 1e-6), the algorithm would
* iterate for a long time trying to find the best solution when indeed
* there is no need to do that if a reasonable threshold has already been
* reached.
* Because of this behaviour the stop threshold can be set to a value much
* lower than the one typically used in RANSAC, and yet the algorithm could
* still produce even smaller thresholds in estimated results.
*/
private double mStopThreshold;
/**
* Quality scores corresponding to each provided homography.
* The larger the score value the better the quality of the sample.
*/
private double[] mQualityScores;
/**
* Constructor.
*/
public PROMedSImageOfAbsoluteConicRobustEstimator() {
super();
mStopThreshold = DEFAULT_STOP_THRESHOLD;
}
/**
* Constructor.
*
* @param listener listener to be notified of events such as when
* estimation starts, ends or its progress significantly changes.
*/
public PROMedSImageOfAbsoluteConicRobustEstimator(
final ImageOfAbsoluteConicRobustEstimatorListener listener) {
super(listener);
mStopThreshold = DEFAULT_STOP_THRESHOLD;
}
/**
* Constructor.
*
* @param homographies list of homographies (2D transformations) used to
* estimate the image of absolute conic (IAC), which can be used to obtain
* pinhole camera intrinsic parameters.
* @throws IllegalArgumentException if not enough homographies are provided
* for default settings. Hence, at least 1 homography must be provided.
*/
public PROMedSImageOfAbsoluteConicRobustEstimator(
final List homographies) {
super(homographies);
mStopThreshold = DEFAULT_STOP_THRESHOLD;
}
/**
* Constructor.
*
* @param homographies list of homographies (2D transformations) used to
* estimate the image of absolute conic (IAC), which can be used to obtain
* pinhole camera intrinsic parameters.
* @param listener listener to be notified of events such as when estimation
* starts, ends or estimation progress changes.
* @throws IllegalArgumentException if not enough homographies are provided
* for default settings. Hence, at least 1 homography must be provided.
*/
public PROMedSImageOfAbsoluteConicRobustEstimator(
final List homographies,
final ImageOfAbsoluteConicRobustEstimatorListener listener) {
super(homographies, listener);
mStopThreshold = DEFAULT_STOP_THRESHOLD;
}
/**
* Constructor.
*
* @param qualityScores quality scores corresponding to each provided
* homography.
* @throws IllegalArgumentException if provided quality scores length is
* smaller than required number of homographies for default settings (i.e.
* 1 homography).
*/
public PROMedSImageOfAbsoluteConicRobustEstimator(final double[] qualityScores) {
this();
internalSetQualityScores(qualityScores);
}
/**
* Constructor.
*
* @param qualityScores quality scores corresponding to each provided
* homography.
* @param listener listener to be notified of events such as when
* estimation starts, ends or its progress significantly changes.
* @throws IllegalArgumentException if provided quality scores length is
* smaller than required number of homographies for default settings (i.e.
* 1 homography).
*/
public PROMedSImageOfAbsoluteConicRobustEstimator(final double[] qualityScores,
final ImageOfAbsoluteConicRobustEstimatorListener listener) {
this(listener);
internalSetQualityScores(qualityScores);
}
/**
* Constructor.
*
* @param homographies list of homographies (2D transformations) used to
* estimate the image of absolute conic (IAC), which can be used to obtain
* pinhole camera intrinsic parameters.
* @param qualityScores quality scores corresponding to each provided
* homography.
* @throws IllegalArgumentException if not enough homographies are provided
* for default settings (i.e. 1 homography) or quality scores and
* homographies don't have the same size.
*/
public PROMedSImageOfAbsoluteConicRobustEstimator(
final List homographies, final double[] qualityScores) {
this(homographies);
internalSetQualityScores(qualityScores);
}
/**
* Constructor.
*
* @param homographies list of homographies (2D transformations) used to
* estimate the image of absolute conic (IAC), which can be used to obtain
* pinhole camera intrinsic parameters.
* @param qualityScores quality scores corresponding to each provided
* homography.
* @param listener listener to be notified of events such as when estimation
* starts, ends or estimation progress changes.
* @throws IllegalArgumentException if not enough homographies are provided
* for default settings (i.e. 1 homography) or quality scores and
* homographies don't have the same size.
*/
public PROMedSImageOfAbsoluteConicRobustEstimator(
final List homographies, final double[] qualityScores,
final ImageOfAbsoluteConicRobustEstimatorListener listener) {
this(homographies, listener);
internalSetQualityScores(qualityScores);
}
/**
* Returns threshold to be used to keep the algorithm iterating in case that
* best estimated threshold using median of residuals is not small enough.
* Once a solution is found that generates a threshold below this value, the
* algorithm will stop.
* The stop threshold can be used to prevent the PROMedS algorithm iterating
* too many times in cases where samples have a very similar accuracy.
* For instance, in cases where proportion of outliers is very small (close
* to 0%), and samples are very accurate (i.e. 1e-6), the algorithm would
* iterate for a long time trying to find the best solution when indeed
* there is no need to do that if a reasonable threshold has already been
* reached.
* Because of this behaviour the stop threshold can be set to a value much
* lower than the one typically used in RANSAC, and yet the algorithm could
* still produce even smaller thresholds in estimated results.
*
* @return stop threshold to stop the algorithm prematurely when a certain
* accuracy has been reached.
*/
public double getStopThreshold() {
return mStopThreshold;
}
/**
* Sets threshold to be used to keep the algorithm iterating in case that
* best estimated threshold using median of residuals is not small enough.
* Once a solution is found that generates a threshold below this value, the
* algorithm will stop.
* The stop threshold can be used to prevent the PROMedS algorithm iterating
* too many times in cases where samples have a very similar accuracy.
* For instance, in cases where proportion of outliers is very small (close
* to 0%), and samples are very accurate (i.e. 1e-6), the algorithm would
* iterate for a long time trying to find the best solution when indeed
* there is no need to do that if a reasonable threshold has already been
* reached.
* Because of this behaviour the stop threshold can be set to a value much
* lower than the one typically used in RANSAC, and yet the algorithm could
* still produce even smaller thresholds in estimated results.
*
* @param stopThreshold stop threshold to stop the algorithm prematurely
* when a certain accuracy has been reached.
* @throws IllegalArgumentException if provided value is zero or negative.
* @throws LockedException if robust estimator is locked because an
* estimation is already in progress.
*/
public void setStopThreshold(final double stopThreshold) throws LockedException {
if (isLocked()) {
throw new LockedException();
}
if (stopThreshold <= MIN_STOP_THRESHOLD) {
throw new IllegalArgumentException();
}
mStopThreshold = stopThreshold;
}
/**
* Returns quality scores corresponding to each provided homography.
* The larger the score value the better the quality of the sampled
* homography
*
* @return quality scores corresponding to each homography.
*/
@Override
public double[] getQualityScores() {
return mQualityScores;
}
/**
* Sets quality scores corresponding to each provided homography.
* The larger the score value the better the quality of the sampled
* homography.
*
* @param qualityScores quality scores corresponding to each homography.
* @throws LockedException if robust estimator is locked because an
* estimation is already in progress.
* @throws IllegalArgumentException if provided quality scores length is
* smaller than minimum required number of homographies (i.e. 1 homography).
*/
@Override
public void setQualityScores(final double[] qualityScores) throws LockedException {
if (isLocked()) {
throw new LockedException();
}
internalSetQualityScores(qualityScores);
}
/**
* Indicates if estimator is ready to start the IAC estimation.
* This is true when input data (i.e. homographies) is provided and list
* contains at least the minimum number of required homographies, and
* also quality scores are provided.
*
* @return true if estimator is ready, false otherwise.
*/
@Override
public boolean isReady() {
return super.isReady() && mQualityScores != null &&
mQualityScores.length == mHomographies.size();
}
/**
* Estimates Image of Absolute Conic (IAC)
*
* @return estimated IAC
* @throws LockedException if robust estimator is locked because an
* estimation is already in progress
* @throws NotReadyException if provided input data is not enough to start
* the estimation
* @throws RobustEstimatorException if estimation fails for any reason
* (i.e. numerical instability, no solution available, etc)
*/
@SuppressWarnings("DuplicatedCode")
@Override
public ImageOfAbsoluteConic estimate() throws LockedException,
NotReadyException, RobustEstimatorException {
if (isLocked()) {
throw new LockedException();
}
if (!isReady()) {
throw new NotReadyException();
}
final PROMedSRobustEstimator innerEstimator =
new PROMedSRobustEstimator<>(
new PROMedSRobustEstimatorListener() {
// subset of homographies picked on each iteration
private final List mSubsetHomographies =
new ArrayList<>();
@Override
public double getThreshold() {
return mStopThreshold;
}
@Override
public int getTotalSamples() {
return mHomographies.size();
}
@Override
public int getSubsetSize() {
return mIACEstimator.getMinNumberOfRequiredHomographies();
}
@Override
public void estimatePreliminarSolutions(final int[] samplesIndices,
final List solutions) {
mSubsetHomographies.clear();
for (final int samplesIndex : samplesIndices) {
mSubsetHomographies.add(mHomographies.get(
samplesIndex));
}
try {
mIACEstimator.setLMSESolutionAllowed(false);
mIACEstimator.setHomographies(mSubsetHomographies);
final ImageOfAbsoluteConic iac = mIACEstimator.estimate();
solutions.add(iac);
} catch (final Exception e) {
// if anything fails, no solution is added
}
}
@Override
public double computeResidual(
final ImageOfAbsoluteConic currentEstimation, final int i) {
return residual(currentEstimation, mHomographies.get(i));
}
@Override
public boolean isReady() {
return PROMedSImageOfAbsoluteConicRobustEstimator.this.isReady();
}
@Override
public void onEstimateStart(
final RobustEstimator estimator) {
if (mListener != null) {
mListener.onEstimateStart(
PROMedSImageOfAbsoluteConicRobustEstimator.this);
}
}
@Override
public void onEstimateEnd(
final RobustEstimator estimator) {
if (mListener != null) {
mListener.onEstimateEnd(
PROMedSImageOfAbsoluteConicRobustEstimator.this);
}
}
@Override
public void onEstimateNextIteration(
final RobustEstimator estimator,
final int iteration) {
if (mListener != null) {
mListener.onEstimateNextIteration(
PROMedSImageOfAbsoluteConicRobustEstimator.this,
iteration);
}
}
@Override
public void onEstimateProgressChange(
final RobustEstimator estimator,
final float progress) {
if (mListener != null) {
mListener.onEstimateProgressChange(
PROMedSImageOfAbsoluteConicRobustEstimator.this,
progress);
}
}
@Override
public double[] getQualityScores() {
return mQualityScores;
}
});
try {
mLocked = true;
innerEstimator.setConfidence(mConfidence);
innerEstimator.setMaxIterations(mMaxIterations);
innerEstimator.setProgressDelta(mProgressDelta);
return innerEstimator.estimate();
} catch (final com.irurueta.numerical.LockedException e) {
throw new LockedException(e);
} catch (final com.irurueta.numerical.NotReadyException e) {
throw new NotReadyException(e);
} finally {
mLocked = false;
}
}
/**
* Returns method being used for robust estimation
*
* @return method being used for robust estimation
*/
@Override
public RobustEstimatorMethod getMethod() {
return RobustEstimatorMethod.PROMedS;
}
/**
* Sets quality scores corresponding to each homography.
* This method is used internally and does not check whether instance is
* locked or not.
*
* @param qualityScores quality scores to be set.
* @throws IllegalArgumentException if provided quality scores length is
* smaller than the minimum number of required homographies for current
* settings.
*/
private void internalSetQualityScores(final double[] qualityScores) {
if (qualityScores.length < mIACEstimator.getMinNumberOfRequiredHomographies()) {
throw new IllegalArgumentException();
}
mQualityScores = qualityScores;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy