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 imagingbook.common.mser; 010 011import ij.process.ImageProcessor; 012import imagingbook.common.filter.linear.GaussianFilter; 013import imagingbook.common.geometry.ellipse.GeometricEllipse; 014import imagingbook.common.geometry.mappings.linear.AffineMapping2D; 015import imagingbook.common.geometry.mappings.linear.Translation2D; 016import imagingbook.common.image.ImageMapper; 017import imagingbook.common.image.interpolation.InterpolationMethod; 018import imagingbook.common.math.Matrix; 019import imagingbook.common.mser.components.Component; 020import org.apache.commons.math3.linear.EigenDecomposition; 021import org.apache.commons.math3.linear.MatrixUtils; 022import org.apache.commons.math3.linear.RealMatrix; 023 024/** 025 * <p> 026 * Provides functionality to extract local affine frames from a given image. Assumes that the specified image has a 027 * white background (255) and black (0) objects. See Section 26.5 of [1] for a detailed description. 028 * </p> 029 * <p> 030 * [1] W. Burger, M.J. Burge, <em>Digital Image Processing – An Algorithmic Introduction</em>, 3rd ed, Springer 031 * (2022). 032 * </p> 033 * 034 * @author WB 035 * @version 2022/11/19 036 * @see AffineMapping2D 037 * @see ImageMapper 038 */ 039public class LocalAffineFrameExtractor { // TODO: needs testing! 040 041 private static InterpolationMethod interpolMethod = InterpolationMethod.Bilinear; 042 043 private final ImageProcessor sourceIp; 044 private final int m; // size of frame to extract (W = H = 2 m + 1) 045 private final double s; // size ratio between outer and inner ellipse 046 private final boolean useAntiAliasingFilter; 047 048 /** 049 * Full constructor. 050 * @param sourceIp the image to extract frames from 051 * @param m size of the (square) frames to extract (W = H = 2 m + 1) 052 * @param s size ratio between outer and inner ellipse 053 * @param useAntiAliasingFilter turn anti-aliasing filter on or off 054 */ 055 public LocalAffineFrameExtractor(ImageProcessor sourceIp, int m, double s, boolean useAntiAliasingFilter) { 056 this.sourceIp = sourceIp; 057 this.m = m; 058 this.s = s; 059 this.useAntiAliasingFilter = useAntiAliasingFilter; 060 } 061 062 /** 063 * Short constructor. 064 * @param sourceIp the image to extract frames from 065 * @param m size of the (square) frames to extract (W = H = 2 m + 1) 066 */ 067 public LocalAffineFrameExtractor(ImageProcessor sourceIp, int m) { 068 this(sourceIp, m, 1.5, true); 069 } 070 071 /** 072 * Extracts a single local-affine frame for the specified MSER component. The extracted image has the same type as 073 * the original source image (passed to the constructor). 074 * 075 * @param mser a MSER component 076 * @return the local-affine frame (instance of {@link ImageProcessor}) 077 */ 078 public ImageProcessor getLocalAffineFrame(Component<MserData> mser) { 079 MserData props = mser.getProperties(); 080 double[] xc = props.getCenter().toDoubleArray(); //{mu10, mu01}; 081 double[][] S = props.getCovarianceMatrix(); 082 083 EigenDecomposition ed = new EigenDecomposition(MatrixUtils.createRealMatrix(S)); 084 RealMatrix V = ed.getV(); 085 RealMatrix D = ed.getD(); 086 087 for (int i = 0; i < 2; i++) { 088 D.setEntry(i, i, Math.sqrt(D.getEntry(i, i))); 089 } 090 091 RealMatrix A = V.multiply(D).scalarMultiply(2.0 * s / m); 092 double detA = Matrix.determinant2x2(A.getData()); 093 094 if (detA < 0) { // reflection 095 RealMatrix reflect = MatrixUtils.createRealMatrix(new double[][] {{1,0},{0,-1}}); 096 A = A.multiply(reflect); // undo reflection!! 097 } 098 099 ImageProcessor ip2 = sourceIp.duplicate(); 100 ip2.invert(); // makes the background black, objects white 101 102 if (useAntiAliasingFilter) { 103 GeometricEllipse ellipse = props.getEllipse(); 104 double dMax = s * ellipse.ra / m; // scale ratio 105 double sigma = 0; 106 if (dMax > 1.5) { // down-scaling, sample width increases by factor dScale 107 sigma = (dMax - 1) * 0.5; // factor 0.5 is ad hoc (full dMax blurs too much) 108 } 109 if (sigma > 0) { 110 GaussianFilter gf = new GaussianFilter(sigma); 111 gf.applyTo(ip2); 112 } 113 } 114 115 int frameSize = 2 * m + 1; 116 ImageProcessor laf = sourceIp.createProcessor(frameSize, frameSize); 117 118 AffineMapping2D T1 = new Translation2D(-m, -m); 119 AffineMapping2D AA = new AffineMapping2D(A.getData()); 120 AffineMapping2D T2 = new Translation2D(xc); 121 122 AffineMapping2D M = T1.concat(AA).concat(T2); 123 ImageMapper mapper = new ImageMapper(M, null, interpolMethod); 124 125 mapper.map(ip2, laf); 126 127 laf.invert(); // makes background white again 128 return laf; 129 } 130 131}