All Downloads are FREE. Search and download functionalities are using the official Maven repository.

boofcv.alg.fiducial.qrcode.QrPose3DUtils Maven / Gradle / Ivy

/*
 * Copyright (c) 2021, Peter Abeles. All Rights Reserved.
 *
 * This file is part of BoofCV (http://boofcv.org).
 *
 * 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 boofcv.alg.fiducial.qrcode;

import boofcv.struct.distort.DoNothing2Transform2_F64;
import boofcv.struct.distort.Point2Transform2_F64;
import boofcv.struct.geo.Point2D3D;
import boofcv.struct.geo.PointIndex2D_F64;
import georegression.struct.point.Point2D_F64;
import georegression.struct.point.Point3D_F64;
import org.jetbrains.annotations.Nullable;

import java.util.ArrayList;
import java.util.List;

/**
 * Utilities when estimating the 3D pose of a QR Code. Each landmark (e.g. corner on a locator pattern) is assigned
 * an ID. Functions are provided for accessing coordinates (of each landmark by ID. 3D coordinates in marker frame
 * have values from -1 to 1 to make it more numerically favorable for linear estimators. This scale offset is easily
 * fixed with a multiplication after the pose has been found.
 *
 * @author Peter Abeles
 */
public class QrPose3DUtils {

	// observed pixel observations with indexes
	public List pixelControl = new ArrayList<>();
	// storage for corner locations in marker reference frame in 3D and normalized image coordinate observations
	public List point23 = new ArrayList<>();
	// storage for corner locations in marker reference frame in 3D
	public List point3D = new ArrayList<>();
	// transform from pixel to normalzied image coordinates
	protected Point2Transform2_F64 pixelToNorm = new DoNothing2Transform2_F64();
	protected Point2Transform2_F64 undistToDist = new DoNothing2Transform2_F64(); // undistorted pixel to distorted

	public QrPose3DUtils() {
		for (int i = 0; i < 12; i++) {
			pixelControl.add(new PointIndex2D_F64(0, 0, i));
			point23.add(new Point2D3D());
			point3D.add(new Point3D_F64());
		}
	}

	/**
	 * Converts the corner observations into {@link PointIndex2D_F64} where observations are in pixels
	 */
	public List getLandmarkByIndex( QrCode qr ) {
		pixelControl.get(0).p.setTo(qr.ppCorner.get(0));
		pixelControl.get(1).p.setTo(qr.ppCorner.get(1));
		pixelControl.get(2).p.setTo(qr.ppCorner.get(2));
		pixelControl.get(3).p.setTo(qr.ppCorner.get(3));

		pixelControl.get(4).p.setTo(qr.ppRight.get(0));
		pixelControl.get(5).p.setTo(qr.ppRight.get(1));
		pixelControl.get(6).p.setTo(qr.ppRight.get(2));
		pixelControl.get(7).p.setTo(qr.ppRight.get(3));

		pixelControl.get(8).p.setTo(qr.ppDown.get(0));
		pixelControl.get(9).p.setTo(qr.ppDown.get(1));
		pixelControl.get(10).p.setTo(qr.ppDown.get(2));
		pixelControl.get(11).p.setTo(qr.ppDown.get(3));

		// put it back into distorted pixels. Required by FiducialDetectorPnP
		for (int i = 0; i < pixelControl.size(); i++) {
			Point2D_F64 p = pixelControl.get(i).p;
			undistToDist.compute(p.x, p.y, p);
		}

		return pixelControl;
	}

	/**
	 * Returns a list of {@link Point2D3D}. The 2D observation is the corner in normalized image coordinates.
	 * The 3D location is the location of the corner in the marker's reference frame
	 *
	 * @param qr The qr code
	 * @return List of corner points in marker frame and normalized image coordinates
	 */
	public List getLandmark2D3D( QrCode qr ) {
		int N = qr.getNumberOfModules();

		setPair(0, 0, 0, N, qr.ppCorner.get(0));
		setPair(1, 0, 7, N, qr.ppCorner.get(1));
		setPair(2, 7, 7, N, qr.ppCorner.get(2));
		setPair(3, 7, 0, N, qr.ppCorner.get(3));

		setPair(4, 0, N - 7, N, qr.ppRight.get(0));
		setPair(5, 0, N, N, qr.ppRight.get(1));
		setPair(6, 7, N, N, qr.ppRight.get(2));
		setPair(7, 7, N - 7, N, qr.ppRight.get(3));

		setPair(8, N - 7, 0, N, qr.ppDown.get(0));
		setPair(9, N - 7, 7, N, qr.ppDown.get(1));
		setPair(10, N, 7, N, qr.ppDown.get(2));
		setPair(11, N, 0, N, qr.ppDown.get(3));

		return point23;
	}

	/**
	 * Location of each corner in the QR Code's reference frame in 3D
	 *
	 * @param version QR Code's version
	 * @return List. Recycled on each call
	 */
	public List getLandmark3D( int version ) {
		int N = QrCode.totalModules(version);

		set3D(0, 0, N, point3D.get(0));
		set3D(0, 7, N, point3D.get(1));
		set3D(7, 7, N, point3D.get(2));
		set3D(7, 0, N, point3D.get(3));

		set3D(0, N - 7, N, point3D.get(4));
		set3D(0, N, N, point3D.get(5));
		set3D(7, N, N, point3D.get(6));
		set3D(7, N - 7, N, point3D.get(7));

		set3D(N - 7, 0, N, point3D.get(8));
		set3D(N - 7, 7, N, point3D.get(9));
		set3D(N, 7, N, point3D.get(10));
		set3D(N, 0, N, point3D.get(11));

		return point3D;
	}

	/**
	 * Specifies PNP parameters for a single feature
	 *
	 * @param which Landmark's index
	 * @param row row in the QR code's grid coordinate system
	 * @param col column in the QR code's grid coordinate system
	 * @param N width of grid
	 * @param pixel observed pixel coordinate of feature
	 */
	private void setPair( int which, int row, int col, int N, Point2D_F64 pixel ) {
		set3D(row, col, N, point23.get(which).location);
		pixelToNorm.compute(pixel.x, pixel.y, point23.get(which).observation);
	}

	/**
	 * Specifies 3D location of landmark in marker coordinate system
	 *
	 * @param row row in the QR code's grid coordinate system
	 * @param col column in the QR code's grid coordinate system
	 * @param N width of grid
	 * @param location (Output) location of feature in marker reference frame
	 */
	private void set3D( int row, int col, int N, Point3D_F64 location ) {
		double _N = N;
		double gridX = 2.0*(col/_N - 0.5);
		double gridY = 2.0*(0.5 - row/_N);

		location.setTo(gridX, gridY, 0);
	}

	/**
	 * Specifies transform from pixel to normalize image coordinates
	 */
	public void setLensDistortion( @Nullable Point2Transform2_F64 pixelToNorm,
								   @Nullable Point2Transform2_F64 undistToDist ) {
		if (pixelToNorm == null || undistToDist == null) {
			this.pixelToNorm = new DoNothing2Transform2_F64();
			this.undistToDist = new DoNothing2Transform2_F64();
		} else {
			this.pixelToNorm = pixelToNorm;
			this.undistToDist = undistToDist;
		}
	}

	public Point2Transform2_F64 getPixelToNorm() {
		return pixelToNorm;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy