001/*******************************************************************************
002 * Permission to use and distribute this software is granted under the BSD 2-Clause
003 * "Simplified" License (see http://opensource.org/licenses/BSD-2-Clause).
004 * Copyright (c) 2016-2023 Wilhelm Burger. All rights reserved.
005 * Visit https://imagingbook.com for additional details.
006 ******************************************************************************/
007package imagingbook.calibration.zhang;
008
009import imagingbook.calibration.zhang.util.MathUtil;
010import imagingbook.common.geometry.basic.Pnt2d;
011import imagingbook.common.geometry.basic.Pnt2d.PntDouble;
012import imagingbook.common.geometry.mappings.Mapping2D;
013import org.apache.commons.math3.linear.RealMatrix;
014
015
016/**
017 * This class represents the 2D geometric transformation for an image taken with some camera A to an image taken with
018 * another camera B.
019 *
020 * @author W. Burger
021 * @version 2016-06-01
022 */
023public class InterCameraMapping implements Mapping2D {
024
025        private final Camera camA, camB;
026        private final RealMatrix Abi;    // inverse of the intrinsic camera b matrix (2 x 3)
027
028        public InterCameraMapping(Camera camA, Camera camB) {
029//              this.isInverseFlag = true;      // maps target -> source
030                this.camA = camA;        // camera A (used to produce the source image)
031                this.camB = camB;        // camera B (determines the geometry the target image)
032                this.Abi = camB.getInverseA();
033        }
034
035        @Override
036        public Pnt2d applyTo(Pnt2d uv) {
037                // (u,v) is an observed sensor point
038                // apply the inverse camera mapping to get the distorted (x,y) point:
039                double[] xy = Abi.operate(MathUtil.toHomogeneous(uv.toDoubleArray()));
040
041                // remove the lens distortion of camera b:
042                double[] xyu = camB.unwarp(xy);
043
044                // apply the lens distortion of camera a:
045                double[] xyd = camA.warp(xyu);
046
047                // apply the (forward) camera mapping to get the undistorted sensor point (u',v'):
048                return PntDouble.from(camA.mapToSensorPlane(xyd));
049        }
050
051}