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.ImagePlus;
012import ij.plugin.filter.PlugInFilter;
013import ij.process.ImageProcessor;
014import imagingbook.common.geometry.basic.Pnt2d;
015import imagingbook.common.geometry.basic.Pnt2d.PntInt;
016import imagingbook.common.geometry.mappings.nonlinear.BilinearMapping2D;
017import imagingbook.common.ij.DialogUtils;
018import imagingbook.common.ij.IjUtils;
019import imagingbook.common.image.ImageMapper;
020import imagingbook.common.image.OutOfBoundsStrategy;
021import imagingbook.common.image.interpolation.InterpolationMethod;
022import imagingbook.core.jdoc.JavaDocHelp;
023import imagingbook.sampleimages.GeneralSampleImage;
024
025/**
026 * <p>
027 * ImageJ plugin, applies a bilinear transformation derived from a pair of quadrilaterals P, Q (for the source and
028 * target image, respectively) to the current image. See Sec. 21.1.4 of [1] for details. Optionally opens a sample image
029 * if no image is currently open.
030 * </p>
031 * <p>
032 * [1] W. Burger, M.J. Burge, <em>Digital Image Processing &ndash; An Algorithmic Introduction</em>, 3rd ed, Springer
033 * (2022).
034 * </p>
035 *
036 * @author WB
037 * @version 2022/11/28
038 * @see ImageMapper
039 * @see BilinearMapping2D
040 */
041public class Map_Bilinear implements PlugInFilter, JavaDocHelp {
042        
043        private static Pnt2d[] P = {            // source quadrilateral
044                        PntInt.from(0, 0),
045                        PntInt.from(400, 0),
046                        PntInt.from(400, 400),
047                        PntInt.from(0, 400)
048        };
049
050        private static Pnt2d[] Q = {            // target quadrilateral
051                        PntInt.from(0, 60),
052                        PntInt.from(400, 20),
053                        PntInt.from(300, 400),
054                        PntInt.from(30, 200)
055        };
056        
057        /**
058         * Constructor, asks to open a predefined sample image if no other image
059         * is currently open.
060         */
061        public Map_Bilinear() {
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                // we want the inverse mapping (Q -> P, so we swap P/Q):
075                BilinearMapping2D mi = BilinearMapping2D.fromPoints(Q, P);
076                new ImageMapper(mi, OutOfBoundsStrategy.ZeroValues, InterpolationMethod.Bicubic).map(ip);
077    }
078}