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}