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

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

Go to download

BoofCV is an open source Java library for real-time computer vision and robotics applications.

The newest version!
/*
 * 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.alg.distort.LensDistortionNarrowFOV;
import boofcv.alg.interpolate.InterpolatePixelDistortS;
import boofcv.alg.interpolate.InterpolatePixelS;
import boofcv.core.image.border.FactoryImageBorder;
import boofcv.factory.interpolate.FactoryInterpolation;
import boofcv.struct.border.BorderType;
import boofcv.struct.distort.Point2Transform2_F32;
import boofcv.struct.image.ImageGray;
import georegression.struct.point.Point2D_F32;
import georegression.struct.point.Point2D_F64;
import georegression.struct.shapes.Polygon2D_F64;
import lombok.Getter;
import org.ddogleg.struct.DogArray_F32;
import org.jetbrains.annotations.Nullable;

/**
 * Reads binary values from the qr code's grid. Top left corner of the qr code is it's origin. +x = right and +y = down
 * same as it is in images.
 *
 * @author Peter Abeles
 */
@SuppressWarnings({"NullAway.Init"})
public class QrCodeBinaryGridReader> {
	// Number of points sampled at each bit
	public static final int BIT_INTENSITY_SAMPLES = 5;

	@Getter QrCodeBinaryGridToPixel transformGrid = new QrCodeBinaryGridToPixel();

	InterpolatePixelS interpolate;
	Point2D_F32 pixel = new Point2D_F32();

	int imageWidth, imageHeight;

	float threshold;

	@Nullable QrCode qr;

	public QrCodeBinaryGridReader( Class imageType ) {
		// use nearest neighbor to avoid shifting the location
		interpolate = FactoryInterpolation.nearestNeighborPixelS(imageType);
		interpolate.setBorder(FactoryImageBorder.single(BorderType.EXTENDED, imageType));
	}

	public void setImage( T image ) {
		interpolate.setImage(image);
		imageWidth = image.width;
		imageHeight = image.height;
	}

	public void setLensDistortion( int width, int height, @Nullable LensDistortionNarrowFOV model ) {
		interpolate = FactoryInterpolation.bilinearPixelS(
				this.interpolate.getImageType().getImageClass(), BorderType.EXTENDED);
		if (model != null) {
			Point2Transform2_F32 u2d = model.distort_F32(true, true);
			this.interpolate = new InterpolatePixelDistortS<>(interpolate, u2d);
		}
	}

	public void setMarker( QrCode qr ) {
		this.qr = qr;
		transformGrid.addAllFeatures(qr);
		transformGrid.removeOutsideCornerFeatures();
		transformGrid.computeTransform();
		threshold = (float)(qr.threshCorner + qr.threshDown + qr.threshRight)/3.0f;
	}

	public void setSquare( Polygon2D_F64 square, float threshold ) {
		this.qr = null;
		transformGrid.setTransformFromSquare(square);
		this.threshold = threshold;
	}

	public void setMarkerUnknownVersion( QrCode qr, float threshold ) {
		this.qr = null;
		transformGrid.setTransformFromLinesSquare(qr);
		this.threshold = threshold;
	}

	public void imageToGrid( float x, float y, Point2D_F32 grid ) {
		transformGrid.imageToGrid(x, y, grid);
	}

	public void imageToGrid( Point2D_F32 pixel, Point2D_F32 grid ) {
		transformGrid.imageToGrid(pixel.x, pixel.y, grid);
	}

	/**
	 * Converts a pixel coordinate into a grid coordinate.
	 */
	public void imageToGrid( Point2D_F64 pixel, Point2D_F64 grid ) {
		transformGrid.imageToGrid(pixel.x, pixel.y, grid);
	}

	public void gridToImage( float row, float col, Point2D_F32 image ) {
		transformGrid.gridToImage(row, col, image);
	}

	public void gridToImage( double row, double col, Point2D_F64 image ) {
		transformGrid.gridToImage((float)row, (float)col, pixel);
		image.x = pixel.x;
		image.y = pixel.y;
	}

	public float read( float row, float col ) {
		transformGrid.gridToImage(row, col, pixel);
		return interpolate.get(pixel.x, pixel.y);
	}

	/**
	 * Reads intensity values around the location of each bit.
	 *
	 * @param row grid row
	 * @param col grid column
	 * @param intensity (Output) Storage for intensity values
	 */
	public void readBitIntensity( int row, int col, DogArray_F32 intensity ) {
		float center = 0.5f;

		transformGrid.gridToImage(row + center - 0.2f, col + center, pixel);
		intensity.add(interpolate.get(pixel.x, pixel.y));
		transformGrid.gridToImage(row + center + 0.2f, col + center, pixel);
		intensity.add(interpolate.get(pixel.x, pixel.y));
		transformGrid.gridToImage(row + center, col + center - 0.2f, pixel);
		intensity.add(interpolate.get(pixel.x, pixel.y));
		transformGrid.gridToImage(row + center, col + center + 0.2f, pixel);
		intensity.add(interpolate.get(pixel.x, pixel.y));
		transformGrid.gridToImage(row + center, col + center, pixel);
		intensity.add(interpolate.get(pixel.x, pixel.y));
	}

	/**
	 * Reads a bit from the qr code's data matrix while adjusting for location distortions using known
	 * feature locations.
	 *
	 * @param row grid row
	 * @param col grid column
	 * @return 0 or 1 value for the bit
	 */
	public int readBit( int row, int col ) {
		// todo use adjustments from near by alignment patterns

		float center = 0.5f;

		transformGrid.gridToImage(row + center - 0.2f, col + center, pixel);
		float pixel01 = interpolate.get(pixel.x, pixel.y);
		transformGrid.gridToImage(row + center + 0.2f, col + center, pixel);
		float pixel21 = interpolate.get(pixel.x, pixel.y);
		transformGrid.gridToImage(row + center, col + center - 0.2f, pixel);
		float pixel10 = interpolate.get(pixel.x, pixel.y);
		transformGrid.gridToImage(row + center, col + center + 0.2f, pixel);
		float pixel12 = interpolate.get(pixel.x, pixel.y);
		transformGrid.gridToImage(row + center, col + center, pixel);
		float pixel00 = interpolate.get(pixel.x, pixel.y);

//		float threshold = this.threshold*1.25f;

		int total = 0;
		if (pixel01 < threshold) total++;
		if (pixel21 < threshold) total++;
		if (pixel10 < threshold) total++;
		if (pixel12 < threshold) total++;
		if (pixel00 < threshold) total++;

		if (total >= 3)
			return 1;
		else
			return 0;

//		float value = (pixel01+pixel21+pixel10+pixel12)*0.25f;
//		value = value*0.5f + pixel00*0.5f;

		// in at least one situation this was found to improve the reading
//		float threshold;
//		if( qr  != null ) {
//			int N = qr.getNumberOfModules();
//			if( row > N/2 ) {
//				if( col < N/2 )
//					threshold = (float)qr.threshDown;
//				else {
//					threshold = (float)(qr.threshDown+qr.threshRight)/2f;
//				}
//			} else if( col < N/2 ) {
//				threshold = (float)qr.threshCorner;
//			} else {
//				threshold = (float)qr.threshRight;
//			}
//		} else {
//			threshold = this.threshold;
//		}

//		if( pixel00 < threshold )
//			return 1;
//		else
//			return 0;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy