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}