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 Calibration_Plugins_1; 008 009import ij.IJ; 010import ij.ImagePlus; 011import ij.gui.GenericDialog; 012import ij.plugin.PlugIn; 013import imagingbook.calibration.zhang.Camera; 014import imagingbook.calibration.zhang.ViewTransform; 015import imagingbook.calibration.zhang.data.CalibrationImage; 016import imagingbook.calibration.zhang.data.ZhangData; 017import imagingbook.common.color.sets.BasicAwtColor; 018import imagingbook.common.geometry.basic.Pnt2d; 019import imagingbook.common.ij.overlay.ColoredStroke; 020import imagingbook.common.ij.overlay.ShapeOverlayAdapter; 021import imagingbook.core.jdoc.JavaDocHelp; 022import imagingbook.core.resource.ImageResource; 023 024import java.awt.Shape; 025import java.awt.geom.Ellipse2D; 026import java.awt.geom.Path2D; 027import java.util.ArrayList; 028import java.util.List; 029 030import static imagingbook.common.ij.DialogUtils.formatText; 031 032 033/** 034 * This plugin projects opens an image stack containing the 5 Zhang test images, then outlines the positions of the 035 * observed image points and finally projects the points of the calibration model using the calculated intrinsic camera 036 * parameters (same for all views) and the extrinsic parameters calculated for each view. All data are part of Zhang's 037 * demo data set that comes with the EasyCalib program. NO CALIBRATION is performed here! 038 * Graphic elements are drawn as non-destructive vector overlays: 039 * <ul> 040 * <li>BLUE circles: observed corner points (used for calibration), </li> 041 * <li>RED markers: the model reference points projected using the documented intrinsic camera and view parameters.</li> 042 * </ul> 043 * LOOK CLOSELY / ZOOM IN! The complete image stack with overlay can be saved as a TIFF file. 044 * 045 * @author WB 046 * @version 2022/12/19 047 */ 048public class Validate_EasyCalib_Data implements PlugIn, JavaDocHelp { 049 050 private static ImageResource resource = CalibrationImage.CalibImageStack; 051 private static BasicAwtColor ProjectedModelColor = BasicAwtColor.Red; 052 private static BasicAwtColor CornerMarkColor = BasicAwtColor.Blue; 053 private static double CornerMarkRadius = 2.0; 054 private static double StrokeWidth = 0.5; 055 056 public void run(String arg0) { 057 ImagePlus testIm = resource.getImagePlus(); 058 if (testIm == null) { 059 IJ.error("Could not open calibration images!"); 060 return; 061 } 062 testIm.show(); 063 064 if (!runDialog()) { 065 return; 066 } 067 068 final int M = testIm.getNSlices(); 069 ShapeOverlayAdapter ola = new ShapeOverlayAdapter(); 070 071 072 Pnt2d[] modelPoints = ZhangData.getModelPoints(); // get the reference model points 073 Camera camReal = ZhangData.getCameraIntrinsics(); // get the (known) camera intrinsics 074 ViewTransform[] viewsReal = ZhangData.getAllViewTransforms(); // get the (known) camera views 075 if (viewsReal.length != M) { 076 IJ.error("Wrong number of view transforms: " + viewsReal.length); 077 return; 078 } 079 080 // project and draw the model's reference squares: 081 ola.setStroke(new ColoredStroke(StrokeWidth, ProjectedModelColor.getColor())); 082 for (int i = 0; i < M; i++) { 083 int sliceNo = i + 1; 084 ola.setStackPosition(sliceNo); 085 Pnt2d[] projPnts = camReal.project(viewsReal[i], modelPoints); 086 //for (Shape s : makeCrossShapes(projPnts)) { 087 for (Shape s : makeQuads(projPnts)) { 088 ola.addShape(s); 089 } 090 } 091 092 // draw the observed image (corner) points as circles: 093 ola.setStroke(new ColoredStroke(StrokeWidth, CornerMarkColor.getColor())); 094 Pnt2d[][] obsPoints = ZhangData.getAllObservedPoints(); 095 for (int i = 0; i < obsPoints.length; i++) { 096 int sliceNo = i + 1; 097 ola.setStackPosition(sliceNo); 098 for (Shape s : makeCircleShapes(obsPoints[i])) { 099 ola.addShape(s); 100 } 101 } 102 103 testIm.setOverlay(ola.getOverlay()); 104 } 105 106 // ---------------------------------------------------------------------- 107 108 private List<Shape> makeQuads(Pnt2d[] pnts) { 109 List<Shape> shapes = new ArrayList<>(pnts.length); 110 // 4 successive points make a quad (projected rectangle) 111 for (int i = 0; i < pnts.length; i += 4) { 112 Path2D path = new Path2D.Double(); 113 path.moveTo(pnts[i].getX(), pnts[i].getY()); 114 for (int j = 1; j < 4; j++) { 115 Pnt2d p = pnts[i + j]; 116 path.lineTo(p.getX(), p.getY()); 117 } 118 path.closePath(); 119 shapes.add(path); 120 } 121 return shapes; 122 } 123 124 private List<Shape> makeCircleShapes(Pnt2d[] pnts) { // , Color lineCol 125 final double r = CornerMarkRadius; 126 final double ofs = 0.5; // pixel offset (elements to be placed at pixel centers) 127 List<Shape> shapes = new ArrayList<>(pnts.length); 128 for (int j = 0; j < pnts.length; j++) { 129 double x = pnts[j].getX(); 130 double y = pnts[j].getY(); 131 // OvalRoi circle = new OvalRoi(x - r + ofs, y - r + ofs, 2 * r, 2 * r) ; 132 Shape circle = new Ellipse2D.Double(x - r, y - r, 2 * r, 2 * r); 133 shapes.add(circle); 134 } 135 return shapes; 136 } 137 138 // ------------------------------------------------------------------------- 139 140 private boolean runDialog() { 141 GenericDialog gd = new GenericDialog(this.getClass().getSimpleName()); 142 gd.addHelp(getJavaDocUrl()); 143 gd.setInsets(0, 0, 0); 144 gd.addMessage(formatText(50, 145 "This plugin only displays the 5 sample view images, marks", 146 "the (given) image corner points and projects the model's", 147 "squares using the given view and camera parameters. No", 148 "calibration is performed!")); 149 150 gd.addEnumChoice("Reference squares color", ProjectedModelColor); 151 gd.addEnumChoice("Corner mark color", CornerMarkColor); 152 gd.addNumericField("Corner mark radius", CornerMarkRadius, 1); 153 154 gd.showDialog(); 155 if (gd.wasCanceled()) 156 return false; 157 158 ProjectedModelColor = gd.getNextEnumChoice(BasicAwtColor.class); 159 CornerMarkColor = gd.getNextEnumChoice(BasicAwtColor.class); 160 CornerMarkRadius = gd.getNextNumber(); 161 return true; 162 } 163 164}