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