function [inliers, f, d] = FundamentalLORANSAC2(pts1h, pts2h, idx, nTrials, threshold)

disType = 'sampson';
integerClass = 'int32';
nPts = size(pts1h,2);
outputClass = 'double';


inliers = false(1, nPts);

maxNTrials = nTrials;
curNTrials = zeros(1, integerClass);
bestNInliers = zeros(1, integerClass);


while curNTrials < maxNTrials
    d = estTFormDistance(disType, pts1h, pts2h, idx, outputClass,...
        integerClass);
    
    [curInliers, curNInliers] = findInliers(d, nPts, threshold);
    
    if bestNInliers < curNInliers
        bestNInliers = curNInliers;
        inliers = curInliers;
        if bestNInliers >= 8
            [curInliers,~] = LocalOptimization(curInliers, pts1h, pts2h, threshold, disType);
            curNInliers = length(curInliers);
            if bestNInliers < curNInliers
                bestNInliers = curNInliers;
                inliers = curInliers;
            end
        end
    end
    curNTrials = curNTrials + 1;
end
f = norm8Point(pts1h(:, inliers), pts2h(:, inliers), outputClass,...
      integerClass);
d = computeDistance(disType, pts1h, pts2h, f);
inliers = find(d<=threshold);
end

function [d, f] = estTFormDistance(disType, pts1h, pts2h, idx,...
  outputClass, integerClass)

indices = cast(randperm(length(idx), 8), integerClass);
indices = idx(indices);
f = norm8Point(pts1h(:, indices), pts2h(:, indices), outputClass,...
  integerClass);
d = computeDistance(disType, pts1h, pts2h, f);
end

function f = norm8Point(pts1h, pts2h, outputClass, integerClass)
% Normalize the points
num = cast(size(pts1h, 2), integerClass);
% [pts1h, t1] = vision.internal.normalizePoints(pts1h, 2, outputClass);
% [pts2h, t2] = vision.internal.normalizePoints(pts2h, 2, outputClass);

[pts1h, T1] = normalise2dpts(pts1h);
[pts2h, T2] = normalise2dpts(pts2h);
pts1h(isnan(pts1h)) = 1;
pts2h(isnan(pts2h)) = 1;

% Compute the constraint matrix
m = coder.nullcopy(zeros(num, 9, outputClass));
for idx = 1: num
  m(idx,:) = [...
    pts1h(1,idx)*pts2h(1,idx), pts1h(2,idx)*pts2h(1,idx), pts2h(1,idx), ...
    pts1h(1,idx)*pts2h(2,idx), pts1h(2,idx)*pts2h(2,idx), pts2h(2,idx), ...
                 pts1h(1,idx),              pts1h(2,idx), 1];
end

% Find out the eigen-vector corresponding to the smallest eigen-value.
[~, ~, vm] = svd(m, 0);
f = reshape(vm(:, end), 3, 3)';

% Enforce rank-2 constraint
[u, s, v] = svd(f);
s(end) = 0;
f = u * s * v';

f = f/norm(f(:));

f = T2'*f*T1;
end

function [inliers, nInliers] = findInliers(distance, nPts, threshold)
inliers = find(distance <= threshold);
nInliers = cast(length(inliers), 'like', nPts);
end

function d = computeDistance(disType, pts1h, pts2h, f)
pfp = (pts2h' * f)';
pfp = pfp .* pts1h;
d = sum(pfp, 1) .^ 2;

if strcmp(disType, 'sampson')
  epl1 = f * pts1h;
  epl2 = f' * pts2h;
  d = d ./ (epl1(1,:).^2 + epl1(2,:).^2 + epl2(1,:).^2 + epl2(2,:).^2);
end
d = sqrt(d);
end

function [bestInliers,bestf] = LocalOptimization(curInliers, pts1h, pts2h, t, disType)

integerClass = 'int32';
outputClass = 'double';

f = norm8Point(pts1h(:, curInliers), pts2h(:, curInliers), outputClass,...
    integerClass);

bestInliers = curInliers;
bestNInliers = length(bestInliers);
bestf = f;

s = 8;

th_multiplier = 4*sqrt(2); 
irlsSteps = 10;

% th_multiplier = 2; 
% irlsSteps = 4;

th_step_size = (th_multiplier*t - t)./irlsSteps;
for loirls = 0:irlsSteps
    d = computeDistance(disType, pts1h, pts2h, f);
    loind = find(d<=(th_multiplier*t - th_step_size*loirls));
    if length(loind)>=8
    loind2 = randsample(loind, min(s*7, length(loind)));
    % loind2 = randsample(loind, min(15, length(loind)));
    w = 1./(1+3*d(loind2)/t);
    f = weightedNorm8Point(pts1h(:, loind2), pts2h(:, loind2), w, outputClass, integerClass);
    d = computeDistance(disType, pts1h, pts2h, f);
    loind = find(d<=t);
    if length(loind)>bestNInliers
        bestInliers = loind;
        bestNInliers = length(bestInliers);
        bestf = f;
    end
    end
end


end

function f = weightedNorm8Point(pts1h, pts2h, w, outputClass, integerClass)

num = cast(size(pts1h, 2), integerClass);

[pts1h, T1] = normalise2dpts(pts1h);
[pts2h, T2] = normalise2dpts(pts2h);
pts1h(isnan(pts1h)) = 1;
pts2h(isnan(pts2h)) = 1;

% Compute the constraint matrix
m = coder.nullcopy(zeros(num, 9, outputClass));
for idx = 1: num
  m(idx,:) = [...
    pts1h(1,idx)*pts2h(1,idx), pts1h(2,idx)*pts2h(1,idx), pts2h(1,idx), ...
    pts1h(1,idx)*pts2h(2,idx), pts1h(2,idx)*pts2h(2,idx), pts2h(2,idx), ...
                 pts1h(1,idx),              pts1h(2,idx), 1];
end

m = diag(sqrt(w))*m;
% Find out the eigen-vector corresponding to the smallest eigen-value.
[~, ~, vm] = svd(m, 0);
f = reshape(vm(:, end), 3, 3)';

% Enforce rank-2 constraint
[u, s, v] = svd(f);
s(end) = 0;
f = u * s * v';

% Normalize the fundamental matrix.
f = f / norm(f(:));
f = T2'*f*T1;
end