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 &ndash; 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}