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 – 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}