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 imagingbook.common.ij.IjUtils;
015import imagingbook.common.image.PixelPack;
016import imagingbook.common.math.Matrix;
017
018import static imagingbook.common.math.Arithmetic.sqr;
019
020/**
021 * <p>
022 * Monochromatic edge detector, applicable to color images ({@link ColorProcessor}) only. See Sec. 16.2 (Alg. 16.1) of
023 * [1] for additional details.
024 * </p>
025 * <p>
026 * [1] W. Burger, M.J. Burge, <em>Digital Image Processing &ndash; An Algorithmic Introduction</em>, 3rd ed, Springer
027 * (2022).
028 * </p>
029 *
030 * @author WB
031 * @version 2022/09/11
032 */
033public class MonochromaticEdgeDetector implements EdgeDetector {
034
035        private final int M;    // image width
036        private final int N;    // image height
037        private final FloatProcessor Emag;      // edge magnitude map
038        private final FloatProcessor Eort;      // edge orientation map
039
040        // Sobel-kernels for x/y-derivatives:
041    private static final float[][] HxS = Matrix.multiply(1.0f/8, new float[][] {
042                        {-1, 0, 1},
043                    {-2, 0, 2},
044                    {-1, 0, 1}});
045
046    private static final float[][] HyS = Matrix.multiply(1.0f/8, new float[][] {
047                        {-1, -2, -1},
048                        { 0,  0,  0},
049                        { 1,  2,  1}});
050
051        
052        public MonochromaticEdgeDetector(ColorProcessor cp) {
053                this.M = cp.getWidth();
054                this.N = cp.getHeight();
055                this.Emag = new FloatProcessor(M, N);
056                this.Eort = new FloatProcessor(M, N);
057                findEdges(cp);
058        }
059        
060        private void findEdges(ColorProcessor cp) {
061                FloatProcessor[] I = new PixelPack(cp).getFloatProcessors();
062            FloatProcessor[] Ix = new FloatProcessor[3];
063            FloatProcessor[] Iy = new FloatProcessor[3];
064            
065                for (int k = 0; k < 3; k++) {
066                        Ix[k] = I[k];
067                        Iy[k] = (FloatProcessor) Ix[k].duplicate();
068                        // TODO: convert to generic filter?
069                        IjUtils.convolve(Ix[k], HxS);
070                        IjUtils.convolve(Iy[k], HyS);
071                }
072
073                for (int v = 0; v < N; v++) {
074                        for (int u = 0; u < M; u++) {
075                                // extract the gradients of the R, G, B channels:
076                                double rx = Ix[0].getf(u, v), ry = Iy[0].getf(u, v);
077                                double gx = Ix[1].getf(u, v), gy = Iy[1].getf(u, v);
078                                double bx = Ix[2].getf(u, v), by = Iy[2].getf(u, v);
079                                
080                                double er2 = sqr(rx) + sqr(ry);
081                                double eg2 = sqr(gx) + sqr(gy);
082                                double eb2 = sqr(bx) + sqr(by);
083                                
084                                // assign local edge magnitude:
085                                Emag.setf(u, v, (float) Math.sqrt(er2 + eg2 + eb2));
086                                
087                                // find the maximum gradient channel:
088                                double e2max = er2, cx = rx, cy = ry;   // assume red is the max channel                        
089                                if (eg2 > e2max) {
090                                        e2max = eg2; cx = gx; cy = gy;          // green is the max channel
091                                }
092                                if (eb2 > e2max) {
093                                        e2max = eb2; cx = bx; cy = by;          // blue is the max channel
094                                }
095                                
096                                // calculate edge orientation angle for the maximum channel:
097                                Eort.setf(u, v, (float) Math.atan2(cy, cx));
098                        }
099                }
100        }
101        
102        @Override
103        public FloatProcessor getEdgeMagnitude() {
104                return Emag;
105        }
106
107        @Override
108        public FloatProcessor getEdgeOrientation() {
109                return Eort;
110        }
111        
112}