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 &ndash; 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}