package gdsc.smlm.ij.plugins; /*----------------------------------------------------------------------------- * GDSC SMLM Software * * Copyright (C) 2013 Alex Herbert * Genome Damage and Stability Centre * University of Sussex, UK * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 3 of the License, or * (at your option) any later version. *---------------------------------------------------------------------------*/ import gdsc.smlm.engine.DataFilter; import gdsc.smlm.engine.DataFilterType; import gdsc.smlm.engine.FitEngine; import gdsc.smlm.engine.FitEngineConfiguration; import gdsc.smlm.engine.FitJob; import gdsc.smlm.fitting.FitConfiguration; import gdsc.smlm.fitting.FitFunction; import gdsc.smlm.fitting.FitSolver; import gdsc.smlm.function.gaussian.Gaussian2DFunction; import gdsc.smlm.ij.IJImageSource; import gdsc.smlm.ij.results.ResultsTable; import gdsc.smlm.ij.settings.GlobalSettings; import gdsc.smlm.ij.settings.PSFEstimatorSettings; import gdsc.smlm.ij.settings.ResultsSettings; import gdsc.smlm.ij.settings.SettingsManager; import gdsc.smlm.ij.utils.ImageConverter; import gdsc.core.ij.Utils; import gdsc.smlm.results.AggregatedImageSource; import gdsc.smlm.results.Calibration; import gdsc.smlm.results.ImageSource; import gdsc.smlm.results.InterlacedImageSource; import gdsc.smlm.results.PeakResult; import gdsc.smlm.results.PeakResults; import gdsc.core.utils.Random; import gdsc.core.utils.StoredDataStatistics; import ij.IJ; import ij.ImagePlus; import ij.ImageStack; import ij.gui.GenericDialog; import ij.gui.Roi; import ij.plugin.WindowOrganiser; import ij.plugin.filter.PlugInFilter; import ij.process.ImageProcessor; import ij.text.TextWindow; import java.awt.Color; import java.awt.Component; import java.awt.GridBagConstraints; import java.awt.GridBagLayout; import java.awt.Rectangle; import java.util.Arrays; import java.util.Collection; import org.apache.commons.math3.stat.descriptive.DescriptiveStatistics; import org.apache.commons.math3.stat.descriptive.StatisticalSummary; import org.apache.commons.math3.stat.descriptive.SummaryStatistics; import org.apache.commons.math3.stat.inference.TestUtils; /** * Iteratively fits local maxima using a 2D Gaussian until the PSF converges. */ public class PSFEstimator implements PlugInFilter, PeakResults { private static final String TITLE = "PSF Estimator"; private static TextWindow resultsWindow = null; private double initialPeakStdDev0 = 1; private double initialPeakStdDev1 = 1; private double initialPeakAngle = 0; private boolean extraOptions; private static int optionIntegrateFrames = 1; private int integrateFrames = 1; private static boolean optionInterlacedData = false; private static int optionDataStart = 1; private static int optionDataBlock = 1; private static int optionDataSkip = 0; private boolean interlacedData = false; private int dataStart = 1; private int dataBlock = 1; private int dataSkip = 0; private GlobalSettings globalSettings; private FitEngineConfiguration config; private PSFEstimatorSettings settings; private int flags = DOES_16 | DOES_8G | DOES_32 | NO_CHANGES; private ImagePlus imp; // Required for the significance tests private static final int ANGLE = 0; private static final int X = 1; private static final int Y = 2; private static final int XY = 3; private static final String[] NAMES = { "Angle", "X SD", "Y SD" }; DescriptiveStatistics[] sampleNew = new DescriptiveStatistics[3]; DescriptiveStatistics[] sampleOld = new DescriptiveStatistics[3]; boolean[] ignore = new boolean[3]; public PSFEstimator() { } /* * (non-Javadoc) * * @see ij.plugin.filter.PlugInFilter#setup(java.lang.String, ij.ImagePlus) */ public int setup(String arg, ImagePlus imp) { SMLMUsageTracker.recordPlugin(this.getClass(), arg); extraOptions = Utils.isExtraOptions(); if (imp == null) { IJ.noImage(); return DONE; } globalSettings = SettingsManager.loadSettings(); settings = globalSettings.getPsfEstimatorSettings(); // Reset if (IJ.controlKeyDown()) { config = new FitEngineConfiguration(new FitConfiguration()); globalSettings.setFitEngineConfiguration(config); } else { config = globalSettings.getFitEngineConfiguration(); } Roi roi = imp.getRoi(); if (roi != null && roi.getType() != Roi.RECTANGLE) { IJ.error("Rectangular ROI required"); return DONE; } return showDialog(imp); } /** * @param imp * @return */ private int showDialog(ImagePlus imp) { // Keep class variables for the parameters we are fitting FitConfiguration fitConfig = config.getFitConfiguration(); initialPeakStdDev0 = fitConfig.getInitialPeakStdDev0(); initialPeakStdDev1 = fitConfig.getInitialPeakStdDev1(); initialPeakAngle = fitConfig.getInitialAngle(); if (!extraOptions) { interlacedData = false; integrateFrames = 1; } this.imp = imp; GenericDialog gd = new GenericDialog(TITLE); gd.addHelp(About.HELP_URL); gd.addMessage("Estimate 2D Gaussian to fit maxima"); gd.addNumericField("Initial_StdDev0", initialPeakStdDev0, 3); gd.addNumericField("Initial_StdDev1", initialPeakStdDev1, 3); gd.addNumericField("Initial_Angle", initialPeakAngle, 3); gd.addNumericField("Number_of_peaks", settings.numberOfPeaks, 0); // pValue sets the smallest significance level probability level at which they are said to be different. // i.e. p <= pValue they are different // lower pValue means harder to be found different. // lower pValue means easier to be found the same. gd.addNumericField("p-Value", settings.pValue, 4); gd.addCheckbox("Update_preferences", settings.updatePreferences); gd.addCheckbox("Log_progress", settings.debugPSFEstimator); gd.addCheckbox("Iterate", settings.iterate); gd.addCheckbox("Show_histograms", settings.showHistograms); gd.addNumericField("Histogram_bins", settings.histogramBins, 0); String[] filterTypes = SettingsManager.getNames((Object[]) DataFilterType.values()); gd.addChoice("Spot_filter_type", filterTypes, filterTypes[config.getDataFilterType().ordinal()]); String[] filterNames = SettingsManager.getNames((Object[]) DataFilter.values()); gd.addChoice("Spot_filter", filterNames, filterNames[config.getDataFilter(0).ordinal()]); gd.addSlider("Smoothing", 0, 2.5, config.getSmooth(0)); gd.addSlider("Search_width", 0.5, 2.5, config.getSearch()); gd.addSlider("Border", 0.5, 2.5, config.getBorder()); gd.addSlider("Fitting_width", 2, 4.5, config.getFitting()); if (extraOptions) { gd.addCheckbox("Interlaced_data", optionInterlacedData); gd.addSlider("Integrate_frames", 1, 5, optionIntegrateFrames); } gd.addMessage("--- Gaussian fitting ---"); Component splitLabel = gd.getMessage(); String[] solverNames = SettingsManager.getNames((Object[]) FitSolver.values()); gd.addChoice("Fit_solver", solverNames, solverNames[fitConfig.getFitSolver().ordinal()]); String[] functionNames = SettingsManager.getNames((Object[]) FitFunction.values()); gd.addChoice("Fit_function", functionNames, functionNames[fitConfig.getFitFunction().ordinal()]); // Parameters specific to each Fit solver are collected in a second dialog gd.addNumericField("Fail_limit", config.getFailuresLimit(), 0); gd.addCheckbox("Include_neighbours", config.isIncludeNeighbours()); gd.addSlider("Neighbour_height", 0.01, 1, config.getNeighbourHeightThreshold()); gd.addSlider("Residuals_threshold", 0.01, 1, config.getResidualsThreshold()); gd.addMessage("--- Peak filtering ---\nDiscard fits that shift; are too low; or expand/contract"); gd.addCheckbox("Smart_filter", fitConfig.isSmartFilter()); gd.addCheckbox("Disable_simple_filter", fitConfig.isDisableSimpleFilter()); gd.addSlider("Shift_factor", 0.01, 2, fitConfig.getCoordinateShiftFactor()); gd.addNumericField("Signal_strength", fitConfig.getSignalStrength(), 2); gd.addNumericField("Min_photons", fitConfig.getMinPhotons(), 0); gd.addSlider("Min_width_factor", 0, 0.99, fitConfig.getMinWidthFactor()); gd.addSlider("Width_factor", 1.01, 5, fitConfig.getWidthFactor()); gd.addNumericField("Precision", fitConfig.getPrecisionThreshold(), 2); if (gd.getLayout() != null) { GridBagLayout grid = (GridBagLayout) gd.getLayout(); int xOffset = 0, yOffset = 0; int lastY = -1, rowCount = 0; for (Component comp : gd.getComponents()) { // Check if this should be the second major column if (comp == splitLabel) { xOffset += 2; yOffset -= rowCount; } // Reposition the field GridBagConstraints c = grid.getConstraints(comp); if (lastY != c.gridy) rowCount++; lastY = c.gridy; c.gridx = c.gridx + xOffset; c.gridy = c.gridy + yOffset; c.insets.left = c.insets.left + 10 * xOffset; c.insets.top = 0; c.insets.bottom = 0; grid.setConstraints(comp, c); } if (IJ.isLinux()) gd.setBackground(new Color(238, 238, 238)); } gd.showDialog(); if (gd.wasCanceled() || !readDialog(gd)) return DONE; return flags; } private boolean readDialog(GenericDialog gd) { initialPeakStdDev0 = gd.getNextNumber(); initialPeakStdDev1 = gd.getNextNumber(); initialPeakAngle = gd.getNextNumber(); settings.numberOfPeaks = (int) gd.getNextNumber(); settings.pValue = gd.getNextNumber(); settings.updatePreferences = gd.getNextBoolean(); settings.debugPSFEstimator = gd.getNextBoolean(); settings.iterate = gd.getNextBoolean(); settings.showHistograms = gd.getNextBoolean(); settings.histogramBins = (int) gd.getNextNumber(); config.setDataFilterType(gd.getNextChoiceIndex()); config.setDataFilter(gd.getNextChoiceIndex(), Math.abs(gd.getNextNumber()), 0); config.setSearch(gd.getNextNumber()); config.setBorder(gd.getNextNumber()); config.setFitting(gd.getNextNumber()); if (extraOptions) { interlacedData = optionInterlacedData = gd.getNextBoolean(); integrateFrames = optionIntegrateFrames = (int) gd.getNextNumber(); } FitConfiguration fitConfig = config.getFitConfiguration(); fitConfig.setFitSolver(gd.getNextChoiceIndex()); fitConfig.setFitFunction(gd.getNextChoiceIndex()); config.setFailuresLimit((int) gd.getNextNumber()); config.setIncludeNeighbours(gd.getNextBoolean()); config.setNeighbourHeightThreshold(gd.getNextNumber()); config.setResidualsThreshold(gd.getNextNumber()); fitConfig.setSmartFilter(gd.getNextBoolean()); fitConfig.setDisableSimpleFilter(gd.getNextBoolean()); fitConfig.setCoordinateShiftFactor(gd.getNextNumber()); fitConfig.setSignalStrength(gd.getNextNumber()); fitConfig.setMinPhotons(gd.getNextNumber()); fitConfig.setMinWidthFactor(gd.getNextNumber()); fitConfig.setWidthFactor(gd.getNextNumber()); fitConfig.setPrecisionThreshold(gd.getNextNumber()); if (gd.invalidNumber()) return false; // Check arguments try { Parameters.isAboveZero("Initial SD0", initialPeakStdDev0); Parameters.isAboveZero("Initial SD1", initialPeakStdDev1); Parameters.isPositive("Initial angle", initialPeakAngle); Parameters.isPositive("Number of peaks", settings.numberOfPeaks); Parameters.isAboveZero("P-value", settings.pValue); Parameters.isEqualOrBelow("P-value", settings.pValue, 0.5); if (settings.showHistograms) Parameters.isAboveZero("Histogram bins", settings.histogramBins); Parameters.isAboveZero("Search width", config.getSearch()); Parameters.isAboveZero("Fitting width", config.getFitting()); Parameters.isPositive("Failures limit", config.getFailuresLimit()); Parameters.isPositive("Neighbour height threshold", config.getNeighbourHeightThreshold()); Parameters.isPositive("Residuals threshold", config.getResidualsThreshold()); Parameters.isPositive("Coordinate Shift factor", fitConfig.getCoordinateShiftFactor()); Parameters.isPositive("Signal strength", fitConfig.getSignalStrength()); Parameters.isPositive("Min photons", fitConfig.getMinPhotons()); Parameters.isPositive("Min width factor", fitConfig.getMinWidthFactor()); Parameters.isPositive("Width factor", fitConfig.getWidthFactor()); } catch (IllegalArgumentException e) { IJ.error(TITLE, e.getMessage()); return false; } if (fitConfig.getFitFunction() != FitFunction.FREE && fitConfig.getFitFunction() != FitFunction.FREE_CIRCULAR && fitConfig.getFitFunction() != FitFunction.CIRCULAR) { String msg = "ERROR: A width-fitting function must be selected (i.e. not fixed-width fitting)"; IJ.error(TITLE, msg); log(msg); return false; } final String filename = SettingsManager.getSettingsFilename(); SettingsManager.saveSettings(globalSettings, filename); if (!PeakFit.configureSmartFilter(globalSettings, filename)) return false; if (!PeakFit.configureDataFilter(globalSettings, filename, false)) return false; if (!PeakFit.configureFitSolver(globalSettings, filename, false)) return false; // Extra parameters are needed for interlaced data if (interlacedData) { gd = new GenericDialog(TITLE); gd.addMessage("Interlaced data requires a repeating pattern of frames to process.\n" + "Describe the regular repeat of the data:\n \n" + "Start = The first frame that contains data\n" + "Block = The number of continuous frames containing data\n" + "Skip = The number of continuous frames to ignore before the next data\n \n" + "E.G. 2:9:1 = Data was imaged from frame 2 for 9 frames, 1 frame to ignore, then repeat."); gd.addNumericField("Start", optionDataStart, 0); gd.addNumericField("Block", optionDataBlock, 0); gd.addNumericField("Skip", optionDataSkip, 0); gd.showDialog(); if (gd.wasCanceled()) return false; if (!gd.wasCanceled()) { dataStart = (int) gd.getNextNumber(); dataBlock = (int) gd.getNextNumber(); dataSkip = (int) gd.getNextNumber(); if (dataStart > 0 && dataBlock > 0 && dataSkip > 0) { // Store options for next time optionInterlacedData = true; optionDataStart = dataStart; optionDataBlock = dataBlock; optionDataSkip = dataSkip; } } else { interlacedData = false; } } return true; } /* * (non-Javadoc) * * @see ij.plugin.filter.PlugInFilter#run(ij.process.ImageProcessor) */ public void run(ImageProcessor ip) { int result; while (true) { result = estimatePSF(); if (settings.iterate && result == TRY_AGAIN) { continue; } break; } if (result < INSUFFICIENT_PEAKS) { log("Finished. Check the table for final parameters"); // Only save if successful if (settings.updatePreferences) SettingsManager.saveSettings(globalSettings); } } private static final int TRY_AGAIN = 0; private static final int COMPLETE = 1; private static final int INSUFFICIENT_PEAKS = 2; private static final int ABORTED = 3; private static final int EXCEPTION = 4; private static final int BAD_ESTIMATE = 5; private int estimatePSF() { log("Estimating PSF ... Press escape to abort"); PeakFit fitter = createFitter(); // Use the fit configuration to generate a Gaussian function to test what is being evaluated Gaussian2DFunction gf = config.getFitConfiguration().createGaussianFunction(1, 1, 1, new double[] { 0, 10, initialPeakAngle, 0, 0, initialPeakStdDev0, initialPeakStdDev1 }); createResultsWindow(); int iteration = 0; ignore[ANGLE] = !gf.evaluatesShape(); ignore[X] = !gf.evaluatesSD0(); ignore[Y] = !gf.evaluatesSD1(); double[] params = new double[] { gf.evaluatesShape() ? initialPeakAngle : 0, gf.evaluatesSD0() ? initialPeakStdDev0 : 0, gf.evaluatesSD1() ? initialPeakStdDev1 : 0, 0, 0 }; double[] params_dev = new double[3]; boolean[] identical = new boolean[4]; double[] p = new double[] { Double.NaN, Double.NaN, Double.NaN, Double.NaN }; addToResultTable(iteration++, 0, params, params_dev, p); if (!calculateStatistics(fitter, params, params_dev)) return (Utils.isInterrupted()) ? ABORTED : INSUFFICIENT_PEAKS; if (!addToResultTable(iteration++, size(), params, params_dev, p)) return BAD_ESTIMATE; boolean tryAgain = false; do { if (!calculateStatistics(fitter, params, params_dev)) return (Utils.isInterrupted()) ? ABORTED : INSUFFICIENT_PEAKS; try { for (int i = 0; i < 3; i++) getP(i, p, identical); if (!ignore[Y]) getPairedP(sampleNew[X], sampleNew[Y], XY, p, identical); if (!addToResultTable(iteration++, size(), params, params_dev, p)) return BAD_ESTIMATE; if ((ignore[ANGLE] || identical[ANGLE] || identical[XY]) && (ignore[X] || identical[X]) && (ignore[Y] || identical[Y])) { tryAgain = checkAngleSignificance() || checkXYSignificance(identical); // Update recommended values. Only use if significant params[X] = sampleNew[X].getMean(); params[Y] = (!ignore[Y] && !identical[XY]) ? sampleNew[Y].getMean() : params[X]; params[ANGLE] = (!ignore[ANGLE]) ? sampleNew[ANGLE].getMean() : 0; // update starting configuration initialPeakAngle = (float) params[ANGLE]; initialPeakStdDev0 = (float) params[X]; initialPeakStdDev1 = (float) params[Y]; if (settings.updatePreferences) { config.getFitConfiguration().setInitialPeakStdDev0((float) params[X]); config.getFitConfiguration().setInitialPeakStdDev1((float) params[Y]); config.getFitConfiguration().setInitialAngle((float) params[ANGLE]); } break; } if (IJ.escapePressed()) { IJ.beep(); IJ.showStatus("Aborted"); return ABORTED; } } catch (Exception e) { e.printStackTrace(); return EXCEPTION; } } while (true); return (tryAgain) ? TRY_AGAIN : COMPLETE; } private boolean checkAngleSignificance() { boolean tryAgain = false; if (ignore[ANGLE]) return tryAgain; // The angle is relative to the major axis (X). // It could be close to 0, 90 or 180 to allow it to be ignored in favour of a free circular function. final double[] angles = sampleNew[ANGLE].getValues(); for (double testAngle : new double[] { 90, 0, 180 }) { // The angle will be in the 0-180 domain. // We need to compute the Statistical summary around the testAngle. StatisticalSummary sampleStats; if (testAngle == 0 || testAngle == 180) { SummaryStatistics stats = new SummaryStatistics(); boolean zeroAngle = (testAngle == 0); for (double a : angles) { if (zeroAngle) { // Convert to -90-90 domain if (a > 90) a -= 180; } else { // Convert to 90-270 domain if (a < 90) a += 180; } stats.addValue(a); } sampleStats = stats; } else { // Already in the 0-180 domain around the angle 90 sampleStats = sampleNew[ANGLE]; } final double p = TestUtils.tTest(testAngle, sampleStats); if (p > settings.pValue) { log("NOTE: Angle is not significant: %g ~ %g (p=%g) => Re-run with fixed zero angle", sampleStats.getMean(), testAngle, p); ignore[ANGLE] = true; config.getFitConfiguration().setFitFunction(FitFunction.FREE_CIRCULAR); tryAgain = true; break; } else debug(" NOTE: Angle is significant: %g !~ %g (p=%g)", sampleNew[ANGLE].getMean(), testAngle, p); } return tryAgain; } private boolean checkXYSignificance(boolean[] identical) { boolean tryAgain = false; if (identical[XY]) { log("NOTE: X-width and Y-width are not significantly different: %g ~ %g => Re-run with circular function", sampleNew[X].getMean(), sampleNew[Y].getMean()); config.getFitConfiguration().setFitFunction(FitFunction.CIRCULAR); tryAgain = true; } return tryAgain; } private void getP(int i, double[] p, boolean[] identical) throws IllegalArgumentException { getP(sampleNew[i], sampleOld[i], i, p, identical); } private void getP(StatisticalSummary sample1, StatisticalSummary sample2, int i, double[] p, boolean[] identical) { if (sample1.getN() < 2) return; // The number returned is the smallest significance level at which one can reject the null // hypothesis that the mean of the paired differences is 0 in favor of the two-sided alternative // that the mean paired difference is not equal to 0. For a one-sided test, divide the returned value by 2 p[i] = TestUtils.tTest(sample1, sample2); identical[i] = (p[i] > settings.pValue); } private void getPairedP(DescriptiveStatistics sample1, DescriptiveStatistics sample2, int i, double[] p, boolean[] identical) throws IllegalArgumentException { if (sample1.getN() < 2) return; // The number returned is the smallest significance level at which one can reject the null // hypothesis that the mean of the paired differences is 0 in favor of the two-sided alternative // that the mean paired difference is not equal to 0. For a one-sided test, divide the returned value by 2 p[i] = TestUtils.pairedTTest(sample1.getValues(), sample2.getValues()); identical[i] = (p[i] > settings.pValue); } private boolean calculateStatistics(PeakFit fitter, double[] params, double[] params_dev) { debug(" Fitting PSF"); swapStatistics(); // Create the fit engine using the PeakFit plugin FitConfiguration fitConfig = config.getFitConfiguration(); fitConfig.setInitialAngle((float) params[0]); fitConfig.setInitialPeakStdDev0((float) params[1]); fitConfig.setInitialPeakStdDev1((float) params[2]); ImageStack stack = imp.getImageStack(); Rectangle roi = stack.getProcessor(1).getRoi(); ImageSource source = new IJImageSource(imp); // Allow interlaced data by wrapping the image source if (interlacedData) { source = new InterlacedImageSource(source, dataStart, dataBlock, dataSkip); } // Allow frame aggregation by wrapping the image source if (integrateFrames > 1) { source = new AggregatedImageSource(source, integrateFrames); } fitter.initialiseImage(source, roi, true); fitter.addPeakResults(this); fitter.initialiseFitting(); FitEngine engine = fitter.createFitEngine(); // Use random slices int[] slices = new int[stack.getSize()]; for (int i = 0; i < slices.length; i++) slices[i] = i + 1; Random rand = new Random(); rand.shuffle(slices); IJ.showStatus("Fitting ..."); // Use multi-threaded code for speed int i; for (i = 0; i < slices.length; i++) { int slice = slices[i]; //debug(" Processing slice = %d\n", slice); IJ.showProgress(size(), settings.numberOfPeaks); ImageProcessor ip = stack.getProcessor(slice); ip.setRoi(roi); // stack processor does not set the bounds required by ImageConverter FitJob job = new FitJob(slice, ImageConverter.getData(ip), roi); engine.run(job); if (sampleSizeReached() || Utils.isInterrupted()) { break; } } if (Utils.isInterrupted()) { IJ.showProgress(1); engine.end(true); return false; } // Wait until we have enough results while (!sampleSizeReached() && !engine.isQueueEmpty()) { IJ.showProgress(size(), settings.numberOfPeaks); try { Thread.sleep(50); } catch (InterruptedException e) { break; } } // End now if we have enough samples engine.end(sampleSizeReached()); IJ.showStatus(""); IJ.showProgress(1); // This count will be an over-estimate given that the provider is ahead of the consumer // in this multi-threaded system debug(" Processed %d/%d slices (%d peaks)", i, slices.length, size()); setParams(ANGLE, params, params_dev, sampleNew[ANGLE]); setParams(X, params, params_dev, sampleNew[X]); setParams(Y, params, params_dev, sampleNew[Y]); if (settings.showHistograms) { int[] idList = new int[NAMES.length]; int count = 0; boolean requireRetile = false; for (int ii = 0; ii < 3; ii++) { if (sampleNew[ii].getN() == 0) continue; StoredDataStatistics stats = new StoredDataStatistics(sampleNew[ii].getValues()); idList[count++] = Utils.showHistogram(TITLE, stats, NAMES[ii], 0, 0, settings.histogramBins, "Mean = " + Utils.rounded(stats.getMean()) + ". Median = " + Utils.rounded(sampleNew[ii].getPercentile(50))); requireRetile = requireRetile || Utils.isNewWindow(); } if (requireRetile && count > 0) { new WindowOrganiser().tileWindows(Arrays.copyOf(idList, count)); } } if (size() < 2) { log("ERROR: Insufficient number of fitted peaks, terminating ..."); return false; } return true; } private void setParams(int i, double[] params, double[] params_dev, DescriptiveStatistics sample) { if (sample.getN() > 0) { params[i] = sample.getMean(); params_dev[i] = sample.getStandardDeviation(); } } private void swapStatistics() { sampleOld[ANGLE] = sampleNew[ANGLE]; sampleOld[X] = sampleNew[X]; sampleOld[Y] = sampleNew[Y]; } private PeakFit createFitter() { ResultsSettings resultsSettings = new ResultsSettings(); resultsSettings.setResultsTable(ResultsTable.NONE); resultsSettings.showDeviations = false; resultsSettings.logProgress = false; resultsSettings.setResultsImage(0); resultsSettings.resultsDirectory = null; PeakFit fitter = new PeakFit(config, resultsSettings, globalSettings.getCalibration()); return fitter; } /** * Create the result window (if it is not available) */ private void createResultsWindow() { if (resultsWindow == null || !resultsWindow.isShowing()) { resultsWindow = new TextWindow(TITLE + " Results", createResultsHeader(), "", 900, 300); } } private String createResultsHeader() { StringBuilder sb = new StringBuilder(); sb.append("Iteration\t"); sb.append("N-peaks\t"); sb.append("Angle\t"); sb.append("+/-\t"); sb.append("p(Angle same)\t"); sb.append("X SD\t"); sb.append("+/-\t"); sb.append("p(X same)\t"); sb.append("Y SD\t"); sb.append("+/-\t"); sb.append("p(Y same)\t"); sb.append("p(XY same)\t"); return sb.toString(); } private boolean addToResultTable(int iteration, int n, double[] params, double[] params_dev, double[] p) { StringBuilder sb = new StringBuilder(); sb.append(iteration).append("\t").append(n).append("\t"); for (int i = 0; i < 3; i++) { sb.append(params[i]).append("\t"); sb.append(params_dev[i]).append("\t"); sb.append(p[i]).append("\t"); } sb.append(p[XY]).append("\t"); resultsWindow.append(sb.toString()); if (params[X] > imp.getWidth() || params[Y] > imp.getWidth()) { log("ERROR: Bad width estimation (try altering the peak validation parameters), terminating ..."); return false; } return true; } private void debug(String format, Object... args) { if (settings.debugPSFEstimator) log(format, args); } private void log(String format, Object... args) { IJ.log(String.format(format, args)); } public void begin() { sampleNew[ANGLE] = new DescriptiveStatistics(); sampleNew[X] = new DescriptiveStatistics(); sampleNew[Y] = new DescriptiveStatistics(); } /* * (non-Javadoc) * * @see gdsc.utils.fitting.results.PeakResults#add(int, int, int, float, double, float, float[], float[]) */ public synchronized void add(int peak, int origX, int origY, float origValue, double chiSquared, float noise, float[] params, float[] paramsStdDev) { if (!sampleSizeReached()) { if (!ignore[ANGLE]) sampleNew[ANGLE].addValue(params[2]); //if (!ignore[X]) sampleNew[X].addValue(params[5]); if (!ignore[Y]) sampleNew[Y].addValue(params[6]); } } private boolean sampleSizeReached() { return size() >= settings.numberOfPeaks; } public synchronized void addAll(Collection<PeakResult> results) { for (PeakResult result : results) add(result.getFrame(), result.origX, result.origY, result.origValue, result.error, result.noise, result.params, result.paramsStdDev); } public int size() { return (int) sampleNew[X].getN(); } public void end() { // Do nothing } /* * (non-Javadoc) * * @see gdsc.utils.fitting.results.PeakResults#isActive() */ public boolean isActive() { return true; } public void setSource(String source) { // Ignored } public ImageSource getSource() { // Ignored return null; } public void setBounds(Rectangle bounds) { // Ignored } public Rectangle getBounds() { // Ignored return null; } public void setCalibration(Calibration calibration) { // Ignored } public Calibration getCalibration() { // Ignored return null; } public void setConfiguration(String configuration) { // Ignored } public String getConfiguration() { // Ignored return null; } public void copySettings(PeakResults peakResults) { // Ignored } public void setSource(ImageSource source) { // Ignored } public String getName() { // Ignored return null; } public void setName(String name) { // Ignored } }