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