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 ******************************************************************************/
009
010package imagingbook.common.edges;
011
012import ij.process.ColorProcessor;
013import ij.process.FloatProcessor;
014import ij.process.ImageProcessor;
015import imagingbook.common.ij.IjUtils;
016import imagingbook.common.math.Matrix;
017
018import static imagingbook.common.math.Arithmetic.sqr;
019import static java.lang.Math.atan2;
020import static java.lang.Math.sqrt;
021
022/**
023 * <p>
024 * Simple grayscale edge detector for all types of images. Color images are converted to grayscale before edge
025 * detection. See Sec. 5.3 of [1] for a detailed description.
026 * </p>
027 * <p>
028 * [1] W. Burger, M.J. Burge, <em>Digital Image Processing &ndash; An Algorithmic Introduction</em>, 3rd ed, Springer
029 * (2022).
030 * </p>
031 *
032 * @author WB
033 * @version 2022/12/12
034 */
035public class GrayscaleEdgeDetector implements EdgeDetector {
036
037        private final int M;    // image width
038        private final int N;    // image height
039        private final FloatProcessor Emag;      // edge magnitude map
040        private final FloatProcessor Eort;      // edge orientation map
041        
042        // Sobel-kernels for x/y-derivatives:
043    private static final float[][] HxS = Matrix.multiply(1.0f/8, new float[][] {
044                        {-1, 0, 1},
045                    {-2, 0, 2},
046                    {-1, 0, 1}});
047
048    private static final float[][] HyS = Matrix.multiply(1.0f/8, new float[][] {
049                        {-1, -2, -1},
050                        { 0,  0,  0},
051                        { 1,  2,  1}});
052
053        public GrayscaleEdgeDetector(ImageProcessor ip) {
054                this.M = ip.getWidth();
055                this.N = ip.getHeight();
056                Emag = new FloatProcessor(M, N);
057                Eort = new FloatProcessor(M, N);
058                findEdges(ip);
059        }
060
061        private void findEdges(ImageProcessor ip) {
062                FloatProcessor I = (ip instanceof ColorProcessor) ?
063                                IjUtils.toFloatProcessor((ColorProcessor) ip) :
064                                ip.convertToFloatProcessor();
065                
066            FloatProcessor Ix = I;
067            FloatProcessor Iy = (FloatProcessor) Ix.duplicate();
068
069                IjUtils.convolve(Ix, HxS);
070                IjUtils.convolve(Iy, HyS);
071                
072                for (int v = 0; v < N; v++) {
073                        for (int u = 0; u < M; u++) {
074                                // extract the gradients of the R, G, B channels:
075                                double dx = Ix.getf(u, v);      
076                                double dy = Iy.getf(u, v);
077                                // calculate local edge magnitude:
078                                Emag.setf(u, v, (float) sqrt(sqr(dx) + sqr(dy)));
079                                // calculate edge orientation for the maximum channel:
080                                Eort.setf(u, v, (float) atan2(dy, dx));
081                        }
082                }
083        }
084        
085        @Override
086        public FloatProcessor getEdgeMagnitude() {
087                return Emag;
088        }
089
090        @Override
091        public FloatProcessor getEdgeOrientation() {
092                return Eort;
093        }
094        
095}