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 Ch20_Discrete_Cosine_Transform; 010 011import ij.ImagePlus; 012import ij.gui.GenericDialog; 013import ij.plugin.filter.PlugInFilter; 014import ij.process.FloatProcessor; 015import ij.process.ImageProcessor; 016import imagingbook.common.ij.DialogUtils; 017import imagingbook.common.math.Matrix; 018import imagingbook.core.jdoc.JavaDocHelp; 019import imagingbook.sampleimages.GeneralSampleImage; 020import imagingbook.spectral.dct.Dct2d; 021import imagingbook.spectral.dct.Dct2dDirect; 022import imagingbook.spectral.dct.Dct2dFast; 023 024import static imagingbook.common.ij.IjUtils.noCurrentImage; 025 026/** 027 * <p> 028 * Calculates and displays the 2-dimensional DCT after converting the input image to a float image. of arbitrary size. 029 * Optionally, either a direct DCT or a fast DCT implementation is used, using either {@code float} or {@code double} 030 * data. See Ch. 20 of [1] for additional details. 031 * </p> 032 * <p> 033 * [1] W. Burger, M.J. Burge, <em>Digital Image Processing – An Algorithmic Introduction</em>, 3rd ed, Springer 034 * (2022). 035 * </p> 036 * 037 * @author WB 038 * @version 2022/04/01 039 */ 040public class DCT_2D_Demo implements PlugInFilter, JavaDocHelp { 041 042 private static boolean UseFastMode = true; 043 private static boolean UseDoublePrecision = false; 044 private static boolean ShowLogSpectrum = true; 045 private static boolean ReconstructImage = true; 046 047 private ImagePlus im; 048 private int w, h; 049 private FloatProcessor reconstruction = null; 050 051 /** 052 * Constructor, asks to open a predefined sample image if no other image is currently open. 053 */ 054 public DCT_2D_Demo() { 055 if (noCurrentImage()) { 056 DialogUtils.askForSampleImage(GeneralSampleImage.IrishManor); 057 } 058 } 059 060 @Override 061 public int setup(String arg, ImagePlus im) { 062 this.im = im; 063 return DOES_8G + DOES_32 + NO_CHANGES; 064 } 065 066 @Override 067 public void run(ImageProcessor ip) { 068 if (!runDialog()) 069 return; 070 071 w = ip.getWidth(); 072 h = ip.getHeight(); 073 FloatProcessor fp = ip.convertToFloatProcessor(); 074 FloatProcessor spectrum = (UseDoublePrecision) ? runDouble(fp) : runFloat(fp); 075 076 if (ShowLogSpectrum) { 077 spectrum.abs(); 078 spectrum.add(1.0); 079 spectrum.log(); 080 } 081 082 spectrum.resetMinAndMax(); 083 String name = im.getShortTitle(); 084 String title = (ShowLogSpectrum) ? 085 name + "-DCT (log. spectrum)" : name + "-DCT (spectrum)"; 086 new ImagePlus(title, spectrum).show(); 087 088 // ---------------------------------------------------- 089 090 if (ReconstructImage) { 091 reconstruction.resetMinAndMax(); 092 new ImagePlus(name + "-reconstructed", reconstruction).show(); 093 } 094 095 } 096 097 private FloatProcessor runFloat(FloatProcessor fp) { 098 Dct2d.Float dct = (UseFastMode) ? new Dct2dFast.Float(w, h) : new Dct2dDirect.Float(w, h); 099 float[][] g = fp.getFloatArray(); 100 101 // calculate the forward DCT: 102 dct.forward(g); 103 FloatProcessor spectrum = new FloatProcessor(g); 104 105 if (ReconstructImage) { 106 dct.inverse(g); 107 this.reconstruction = new FloatProcessor(g); 108 } 109 110 return spectrum; 111 } 112 113 private FloatProcessor runDouble(FloatProcessor fp) { 114 Dct2d.Double dct = (UseFastMode) ? new Dct2dFast.Double(w, h) : new Dct2dDirect.Double(w, h); 115 double[][] g = Matrix.toDouble(fp.getFloatArray()); 116 117 // calculate the forward DCT: 118 dct.forward(g); 119 FloatProcessor spectrum = new FloatProcessor(Matrix.toFloat(g)); 120 121 if (ReconstructImage) { 122 dct.inverse(g); 123 this.reconstruction = new FloatProcessor(Matrix.toFloat(g)); 124 } 125 126 return spectrum; 127 } 128 129 // --------------------------------------------------------------- 130 131 private boolean runDialog() { 132 GenericDialog gd = new GenericDialog(getClass().getSimpleName()); 133 gd.addHelp(getJavaDocUrl()); 134 gd.addCheckbox("Use fast transform", UseFastMode); 135 gd.addCheckbox("Use double precision", UseDoublePrecision); 136 gd.addCheckbox("Show absolute/log spectrum", ShowLogSpectrum); 137 gd.addCheckbox("Reconstruct the input image", ReconstructImage); 138 139 gd.showDialog(); 140 if (gd.wasCanceled()) 141 return false; 142 143 UseFastMode = gd.getNextBoolean(); 144 UseDoublePrecision = gd.getNextBoolean(); 145 ShowLogSpectrum = gd.getNextBoolean(); 146 ReconstructImage = gd.getNextBoolean(); 147 return true; 148 } 149 150}