001/*******************************************************************************
002 * This software is provided as a supplement to the authors' textbooks on digital
003 * image processing published by Springer-Verlag in various languages and editions.
004 * Permission to use and distribute this software is granted under the BSD 2-Clause
005 * "Simplified" License (see http://opensource.org/licenses/BSD-2-Clause).
006 * Copyright (c) 2006-2023 Wilhelm Burger, Mark J. Burge. All rights reserved.
007 * Visit https://imagingbook.com for additional details.
008 ******************************************************************************/
009package Ch21_Geometric_Operations;
010
011import ij.IJ;
012import ij.ImagePlus;
013import ij.plugin.filter.PlugInFilter;
014import ij.process.ImageProcessor;
015import imagingbook.common.geometry.basic.Pnt2d;
016import imagingbook.common.geometry.basic.Pnt2d.PntInt;
017import imagingbook.common.geometry.mappings.linear.AffineMapping2D;
018import imagingbook.common.ij.DialogUtils;
019import imagingbook.common.ij.IjUtils;
020import imagingbook.common.image.ImageMapper;
021import imagingbook.common.image.OutOfBoundsStrategy;
022import imagingbook.common.image.interpolation.InterpolationMethod;
023import imagingbook.common.math.Matrix;
024import imagingbook.core.jdoc.JavaDocHelp;
025import imagingbook.sampleimages.GeneralSampleImage;
026
027/**
028 * <p>
029 * ImageJ plugin, applies an affine transformation derived from a pair of triangles P, Q (for the source and target
030 * image, respectively) to the current image. See Sec. 21.1.3 of [1] for details. Optionally opens a sample image if no
031 * image is currently open.
032 * </p>
033 * <p>
034 * [1] W. Burger, M.J. Burge, <em>Digital Image Processing &ndash; An Algorithmic Introduction</em>, 3rd ed, Springer
035 * (2022).
036 * </p>
037 *
038 * @author WB
039 * @version 2022/11/28
040 * @see ImageMapper
041 * @see AffineMapping2D
042 */
043public class Map_Affine_Triangles implements PlugInFilter, JavaDocHelp {
044        
045        private static Pnt2d[] P = {                    // source triangle
046                        PntInt.from(0, 0),
047                        PntInt.from(400, 0),
048                        PntInt.from(400, 400)
049        };
050
051        private static Pnt2d[] Q = {                    // target triangle
052                        PntInt.from(0, 60),
053                        PntInt.from(400, 20),
054                        PntInt.from(300, 400)
055        };
056                
057        /**
058         * Constructor, asks to open a predefined sample image if no other image
059         * is currently open.
060         */
061        public Map_Affine_Triangles() {
062                if (IjUtils.noCurrentImage()) {
063                        DialogUtils.askForSampleImage(GeneralSampleImage.Kepler);
064                }
065        }
066
067    @Override
068        public int setup(String arg, ImagePlus imp) {
069        return DOES_ALL;
070    }
071
072    @Override
073        public void run(ImageProcessor ip) {
074                // inverse mapping (target to source):
075        AffineMapping2D m = AffineMapping2D.fromPoints(P, Q);   // forward mapping P -> Q
076                AffineMapping2D mi = m.getInverse();                                    // inverse mapping Q -> P
077                new ImageMapper(mi, OutOfBoundsStrategy.ZeroValues, InterpolationMethod.Bicubic).map(ip);
078                
079                IJ.log("A = \n" + Matrix.toString(m.getTransformationMatrix()));
080    }
081}