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