package ij.plugin.frame;
import java.awt.*;
import java.awt.event.*;
import java.awt.image.*;
import ij.*;
import ij.plugin.*;
import ij.process.*;
import ij.gui.*;
import ij.measure.*;
import ij.plugin.frame.Recorder;
import ij.plugin.filter.*;
import ij.plugin.ChannelSplitter;
import ij.plugin.Thresholder;
public class ThresholdAdjuster extends PlugInDialog implements PlugIn, Measurements,
Runnable, ActionListener, AdjustmentListener, ItemListener, KeyListener, MouseWheelListener, ImageListener {
public static final String LOC_KEY = "threshold.loc";
public static final String MODE_KEY = "threshold.mode";
public static final String DARK_BACKGROUND = "threshold.dark";
static final int RED=0, BLACK_AND_WHITE=1, OVER_UNDER=2;
static final String[] modes = {"Red","B&W", "Over/Under"};
static final double defaultMinThreshold = 85;
static final double defaultMaxThreshold = 170;
static final int DEFAULT = 0;
static boolean fill1 = true;
static boolean fill2 = true;
static boolean useBW = true;
static boolean backgroundToNaN = true;
static ThresholdAdjuster instance;
static int mode = RED;
static String[] methodNames = AutoThresholder.getMethods();
static String method = methodNames[DEFAULT];
static AutoThresholder thresholder = new AutoThresholder();
ThresholdPlot plot = new ThresholdPlot();
Thread thread;
int minValue = -1;
int maxValue = -1;
int sliderRange = 256;
boolean doAutoAdjust,doReset,doApplyLut,doStateChange,doSet;
Panel panel;
Button autoB, resetB, applyB;
int previousImageID;
int previousImageType;
int previousRoiHashCode;
double previousMin, previousMax;
int previousSlice;
boolean imageWasUpdated;
ImageJ ij;
double minThreshold, maxThreshold; Scrollbar minSlider, maxSlider;
TextField minLabel, maxLabel; Label percentiles;
boolean done;
int lutColor;
Choice methodChoice, modeChoice;
Checkbox darkBackground, stackHistogram;
boolean firstActivation = true;
public ThresholdAdjuster() {
super("Threshold");
ImagePlus cimp = WindowManager.getCurrentImage();
if (cimp!=null && cimp.getBitDepth()==24) {
IJ.error("Threshold Adjuster",
"Image>Adjust>Threshold only works with grayscale images.\n \n"
+"You can:\n"
+" Convert to grayscale: Image>Type>8-bit\n"
+" Convert to RGB stack: Image>Type>RGB Stack\n"
+" Convert to HSB stack: Image>Type>HSB Stack\n"
+" Convert to 3 grayscale images: Image>Color>Split Channels\n"
+" Do color thresholding: Image>Adjust>Color Threshold\n");
return;
}
if (instance!=null) {
instance.firstActivation = true;
instance.toFront();
instance.setup(cimp, true);
return;
}
WindowManager.addWindow(this);
instance = this;
mode = (int)Prefs.get(MODE_KEY, RED);
if (mode<RED || mode>OVER_UNDER) mode = RED;
setLutColor(mode);
IJ.register(PasteController.class);
ij = IJ.getInstance();
Font font = new Font("SansSerif", Font.PLAIN, 10);
GridBagLayout gridbag = new GridBagLayout();
GridBagConstraints c = new GridBagConstraints();
setLayout(gridbag);
int y = 0;
c.gridx = 0;
c.gridy = y++;
c.gridwidth = 2;
c.fill = GridBagConstraints.BOTH;
c.anchor = GridBagConstraints.CENTER;
c.insets = new Insets(10, 10, 0, 10); add(plot, c);
plot.addKeyListener(ij);
c.gridx = 0;
c.gridy = y++;
c.insets = new Insets(1, 10, 0, 10);
percentiles = new Label("");
percentiles.setFont(font);
add(percentiles, c);
minSlider = new Scrollbar(Scrollbar.HORIZONTAL, sliderRange/3, 1, 0, sliderRange);
c.gridx = 0;
c.gridy = y++;
c.gridwidth = 1;
c.weightx = IJ.isMacintosh()?90:100;
c.fill = GridBagConstraints.HORIZONTAL;
c.insets = new Insets(1, 10, 0, 0);
add(minSlider, c);
minSlider.addAdjustmentListener(this);
minSlider.addMouseWheelListener(this);
minSlider.setUnitIncrement(1);
minSlider.setFocusable(false);
c.gridx = 1;
c.gridwidth = 1;
c.weightx = IJ.isMacintosh()?10:0;
c.insets = new Insets(5, 0, 0, 10);
String text = "000000";
int columns = 4;
minLabel = new TextField(text,columns);
minLabel.setFont(font);
add(minLabel, c);
minLabel.addMouseWheelListener(this);
minLabel.addKeyListener(this);
maxSlider = new Scrollbar(Scrollbar.HORIZONTAL, sliderRange*2/3, 1, 0, sliderRange);
c.gridx = 0;
c.gridy = y++;
c.gridwidth = 1;
c.weightx = 100;
c.insets = new Insets(2, 10, 0, 0);
add(maxSlider, c);
maxSlider.addAdjustmentListener(this);
maxSlider.addMouseWheelListener(this);
maxSlider.setUnitIncrement(1);
maxSlider.setFocusable(false);
c.gridx = 1;
c.gridwidth = 1;
c.weightx = 0;
c.insets = new Insets(2, 0, 0, 10);
maxLabel = new TextField(text,columns);
maxLabel.setFont(font);
add(maxLabel, c);
maxLabel.addMouseWheelListener(this);
maxLabel.addKeyListener(this);
panel = new Panel();
methodChoice = new Choice();
for (int i=0; i<methodNames.length; i++)
methodChoice.addItem(methodNames[i]);
methodChoice.select(method);
methodChoice.addItemListener(this);
panel.add(methodChoice);
modeChoice = new Choice();
for (int i=0; i<modes.length; i++)
modeChoice.addItem(modes[i]);
modeChoice.select(mode);
modeChoice.addItemListener(this);
panel.add(modeChoice);
c.gridx = 0;
c.gridy = y++;
c.gridwidth = 2;
c.insets = new Insets(8, 5, 0, 5);
c.anchor = GridBagConstraints.CENTER;
c.fill = GridBagConstraints.NONE;
add(panel, c);
panel = new Panel();
boolean db = Prefs.get(DARK_BACKGROUND, Prefs.blackBackground?true:false);
darkBackground = new Checkbox("Dark background");
darkBackground.setState(db);
darkBackground.addItemListener(this);
panel.add(darkBackground);
stackHistogram = new Checkbox("Stack histogram");
stackHistogram.setState(false);
stackHistogram.addItemListener(this);
panel.add(stackHistogram);
c.gridx = 0;
c.gridy = y++;
c.gridwidth = 2;
c.insets = new Insets(5, 5, 0, 5);
add(panel, c);
int trim = IJ.isMacOSX()?11:0;
panel = new Panel();
autoB = new TrimmedButton("Auto",trim);
autoB.addActionListener(this);
autoB.addKeyListener(ij);
panel.add(autoB);
applyB = new TrimmedButton("Apply",trim);
applyB.addActionListener(this);
applyB.addKeyListener(ij);
panel.add(applyB);
resetB = new TrimmedButton("Reset",trim);
resetB.addActionListener(this);
resetB.addKeyListener(ij);
panel.add(resetB);
c.gridx = 0;
c.gridy = y++;
c.gridwidth = 2;
c.insets = new Insets(0, 5, 10, 5);
add(panel, c);
addKeyListener(ij); pack();
Point loc = Prefs.getLocation(LOC_KEY);
if (loc!=null)
setLocation(loc);
else
GUI.center(this);
if (IJ.isMacOSX()) setResizable(false);
show();
thread = new Thread(this, "ThresholdAdjuster");
thread.start();
ImagePlus imp = WindowManager.getCurrentImage();
ImagePlus.addImageListener(this);
if (imp!=null)
setup(imp, true);
}
public synchronized void adjustmentValueChanged(AdjustmentEvent e) {
if (e.getSource()==minSlider)
minValue = minSlider.getValue();
else
maxValue = maxSlider.getValue();
notify();
}
public synchronized void actionPerformed(ActionEvent e) {
Button b = (Button)e.getSource();
if (b==null) return;
if (b==resetB)
doReset = true;
else if (b==autoB)
doAutoAdjust = true;
else if (b==applyB)
doApplyLut = true;
notify();
}
public synchronized void mouseWheelMoved(MouseWheelEvent e) {
if (e.getSource()==minSlider || e.getSource()==minLabel) {
minSlider.setValue(minSlider.getValue() + e.getWheelRotation());
minValue = minSlider.getValue();
} else {
maxSlider.setValue(maxSlider.getValue() + e.getWheelRotation());
maxValue = maxSlider.getValue();
}
notify();
}
public synchronized void keyPressed(KeyEvent e) {
if (e.getKeyCode() == KeyEvent.VK_ENTER) {
doSet = true;
} else if(e.getKeyCode() == KeyEvent.VK_DOWN) {
if (e.getSource()==minLabel) {
minSlider.setValue(minSlider.getValue() - 1);
minValue = minSlider.getValue();
} else {
maxSlider.setValue(maxSlider.getValue() - 1);
maxValue = maxSlider.getValue();
}
} else if(e.getKeyCode() == KeyEvent.VK_UP) {
if (e.getSource()==minLabel) {
minSlider.setValue(minSlider.getValue() + 1);
minValue = minSlider.getValue();
} else {
maxSlider.setValue(maxSlider.getValue() + 1);
maxValue = maxSlider.getValue();
}
}
notify();
}
public void keyReleased(KeyEvent e) {}
public void keyTyped(KeyEvent e) {}
public void imageUpdated(ImagePlus imp) {
if (imp.getID()==previousImageID && Thread.currentThread()!=thread)
imageWasUpdated = true;
}
public void imageOpened(ImagePlus imp) {}
public void imageClosed(ImagePlus imp) {}
void setLutColor(int mode) {
switch (mode) {
case RED:
lutColor = ImageProcessor.RED_LUT;
break;
case BLACK_AND_WHITE:
lutColor = ImageProcessor.BLACK_AND_WHITE_LUT;
break;
case OVER_UNDER:
lutColor = ImageProcessor.OVER_UNDER_LUT;
break;
}
}
public synchronized void itemStateChanged(ItemEvent e) {
Object source = e.getSource();
if (source==methodChoice) {
method = methodChoice.getSelectedItem();
doAutoAdjust = true;
} else if (source==modeChoice) {
mode = modeChoice.getSelectedIndex();
setLutColor(mode);
doStateChange = true;
if (Recorder.record) {
if (Recorder.scriptMode())
Recorder.recordCall("ThresholdAdjuster.setMode(\""+modes[mode]+"\");");
else
Recorder.recordString("call(\"ij.plugin.frame.ThresholdAdjuster.setMode\", \""+modes[mode]+"\");\n");
}
} else
doAutoAdjust = true;
notify();
}
ImageProcessor setup(ImagePlus imp, boolean enableAutoThreshold) {
if (IJ.debugMode) IJ.log("ThresholdAdjuster.setup: "+enableAutoThreshold);
if (imp==null)
return null;
ImageProcessor ip;
int type = imp.getType();
if (type==ImagePlus.COLOR_RGB || (imp.isComposite()&&((CompositeImage)imp).getMode()==IJ.COMPOSITE))
return null;
ip = imp.getProcessor();
boolean minMaxChange = false;
boolean not8Bits = type==ImagePlus.GRAY16 || type==ImagePlus.GRAY32;
int slice = imp.getCurrentSlice();
if (not8Bits) {
if (ip.getMin()==plot.stackMin && ip.getMax()==plot.stackMax && !imageWasUpdated)
minMaxChange = false;
else if (ip.getMin()!=previousMin || ip.getMax()!=previousMax || imageWasUpdated) {
minMaxChange = true;
previousMin = ip.getMin();
previousMax = ip.getMax();
} else if (slice!=previousSlice)
minMaxChange = true;
}
int id = imp.getID();
int roiHashCode = roiHashCode(imp.getRoi());
if (minMaxChange || id!=previousImageID || type!=previousImageType ||
imageWasUpdated || roiHashCode!=previousRoiHashCode) {
minThreshold = ip.getMinThreshold();
maxThreshold = ip.getMaxThreshold();
boolean isThreshold = minThreshold != ImageProcessor.NO_THRESHOLD
&& ip.getCurrentColorModel() != ip.getColorModel(); if (not8Bits && minMaxChange) {
double max1 = ip.getMax();
ip.resetMinAndMax();
if (maxThreshold==max1)
maxThreshold = ip.getMax();
}
ImageStatistics stats = plot.setHistogram(imp, entireStack(imp));
if (stats == null)
return null;
if (isThreshold) {
minThreshold = scaleDown(ip, minThreshold);
maxThreshold = scaleDown(ip, maxThreshold);
} else {
if (enableAutoThreshold && !isThreshold)
autoSetLevels(ip, stats);
else
minThreshold = ImageProcessor.NO_THRESHOLD; }
scaleUpAndSet(ip, minThreshold, maxThreshold);
updateLabels(imp, ip);
updatePercentiles(imp, ip);
updatePlot(ip);
updateScrollBars();
imp.updateAndDraw();
imageWasUpdated = false;
}
previousImageID = id;
previousImageType = type;
previousRoiHashCode = roiHashCode;
previousSlice = slice;
firstActivation = false;
return ip;
}
boolean entireStack(ImagePlus imp) {
return stackHistogram!=null && stackHistogram.getState() && imp.getStackSize()>1;
}
void autoSetLevels(ImageProcessor ip, ImageStatistics stats) {
if (stats==null || stats.histogram==null) {
minThreshold = defaultMinThreshold;
maxThreshold = defaultMaxThreshold;
return;
}
boolean darkb = darkBackground!=null && darkBackground.getState();
boolean invertedLut = ip.isInvertedLut();
int modifiedModeCount = stats.histogram[stats.mode];
if (!method.equals(methodNames[DEFAULT]))
stats.histogram[stats.mode] = plot.originalModeCount;
int threshold = thresholder.getThreshold(method, stats.histogram);
stats.histogram[stats.mode] = modifiedModeCount;
if (darkb) {
if (invertedLut)
{minThreshold=0; maxThreshold=threshold;}
else
{minThreshold=threshold+1; maxThreshold=255;}
} else {
if (invertedLut)
{minThreshold=threshold+1; maxThreshold=255;}
else
{minThreshold=0; maxThreshold=threshold;}
}
if (minThreshold>255)
minThreshold = 255;
if (Recorder.record) {
boolean stack = stackHistogram!=null && stackHistogram.getState();
String options = method+(darkb?" dark":"")+(stack?" stack":"");
if (Recorder.scriptMode())
Recorder.recordCall("IJ.setAutoThreshold(imp, \""+options+"\");");
else
Recorder.record("setAutoThreshold", options);
}
}
void scaleUpAndSet(ImageProcessor ip, double lower, double upper) {
ip.scaleAndSetThreshold(lower, upper, lutColor);
}
double scaleDown(ImageProcessor ip, double threshold) {
if (ip instanceof ByteProcessor)
return threshold;
double min = ip.getMin();
double max = ip.getMax();
if (max>min)
return ((threshold-min)/(max-min))*255.0;
else
return ImageProcessor.NO_THRESHOLD;
}
double scaleUp(ImageProcessor ip, double threshold) {
double min = ip.getMin();
double max = ip.getMax();
if (max>min)
return min + (threshold/255.0)*(max-min);
else
return ImageProcessor.NO_THRESHOLD;
}
void updatePlot(ImageProcessor ip) {
int min = (int)Math.round(minThreshold);
if (min<0) min=0;
if (min>255) min=255;
if (ip.getMinThreshold()==ImageProcessor.NO_THRESHOLD)
min = -1;
int max = (int)Math.round(maxThreshold);
if (max<0) max=0;
if (max>255) max=255;
plot.lowerThreshold = min;
plot.upperThreshold = max;
plot.mode = mode;
plot.repaint();
}
void updatePercentiles(ImagePlus imp, ImageProcessor ip) {
if (percentiles==null)
return;
ImageStatistics stats = plot.stats;
int minThresholdInt = (int)Math.round(minThreshold);
if (minThresholdInt<0) minThresholdInt=0;
if (minThresholdInt>255) minThresholdInt=255;
int maxThresholdInt = (int)Math.round(maxThreshold);
if (maxThresholdInt<0) maxThresholdInt=0;
if (maxThresholdInt>255) maxThresholdInt=255;
if (stats!=null && stats.histogram!=null && stats.histogram.length==256
&& ip.getMinThreshold()!=ImageProcessor.NO_THRESHOLD) {
int[] histogram = stats.histogram;
int below = 0, inside = 0, above = 0;
int minValue=0, maxValue=255;
if (imp.getBitDepth()==16 && !entireStack(imp)) {
ip.setRoi(imp.getRoi());
histogram = ip.getHistogram();
minThresholdInt = (int)Math.round(ip.getMinThreshold());
if (minThresholdInt<0) minThresholdInt=0;
maxThresholdInt = (int)Math.round(ip.getMaxThreshold());
if (maxThresholdInt>65535) maxThresholdInt=65535;
minValue=(int)ip.getMin(); maxValue=(int)ip.getMax();
}
for (int i=minValue; i<minThresholdInt; i++)
below += histogram[i];
for (int i=minThresholdInt; i<=maxThresholdInt; i++)
inside += histogram[i];
for (int i=maxThresholdInt+1; i<=maxValue; i++)
above += histogram[i];
int total = below + inside + above;
int digits = imp.getCalibration()!=null || (ip instanceof FloatProcessor) ?
Math.max(Analyzer.getPrecision(), 2) : 0;
if (mode==OVER_UNDER)
percentiles.setText("below: "+IJ.d2s(100.*below/total)+" %, above: "+IJ.d2s(100.*above/total)+" %");
else
percentiles.setText(IJ.d2s(100.*inside/total)+" %");
} else
percentiles.setText("");
}
void updateLabels(ImagePlus imp, ImageProcessor ip) {
if (minLabel==null || maxLabel==null)
return;
double min = ip.getMinThreshold();
double max = ip.getMaxThreshold();
if (min==ImageProcessor.NO_THRESHOLD) {
minLabel.setText("");
maxLabel.setText("");
} else {
Calibration cal = imp.getCalibration();
if (cal.calibrated()) {
min = cal.getCValue((int)min);
max = cal.getCValue((int)max);
}
if (((int)min==min && (int)max==max) || (ip instanceof ShortProcessor) || max>99999.0) {
minLabel.setText(ResultsTable.d2s(min,0));
maxLabel.setText(ResultsTable.d2s(max,0));
} else {
minLabel.setText(""+(min<-3.4e38?"-3.4e38":ResultsTable.d2s(min,2)));
maxLabel.setText(""+ResultsTable.d2s(max,max==Double.MAX_VALUE?0:2));
}
}
}
void updateScrollBars() {
minSlider.setValue((int)minThreshold);
maxSlider.setValue((int)maxThreshold);
}
void doMasking(ImagePlus imp, ImageProcessor ip) {
ImageProcessor mask = imp.getMask();
if (mask!=null)
ip.reset(mask);
}
void adjustMinThreshold(ImagePlus imp, ImageProcessor ip, double value) {
if (IJ.altKeyDown() || IJ.shiftKeyDown() ) {
double width = maxThreshold-minThreshold;
if (width<1.0) width = 1.0;
minThreshold = value;
maxThreshold = minThreshold+width;
if ((minThreshold+width)>255) {
minThreshold = 255-width;
maxThreshold = minThreshold+width;
minSlider.setValue((int)minThreshold);
}
maxSlider.setValue((int)maxThreshold);
scaleUpAndSet(ip, minThreshold, maxThreshold);
return;
}
minThreshold = value;
if (maxThreshold<minThreshold) {
maxThreshold = minThreshold;
maxSlider.setValue((int)maxThreshold);
}
scaleUpAndSet(ip, minThreshold, maxThreshold);
}
void adjustMaxThreshold(ImagePlus imp, ImageProcessor ip, int cvalue) {
maxThreshold = cvalue;
if (minThreshold>maxThreshold) {
minThreshold = maxThreshold;
minSlider.setValue((int)minThreshold);
}
if (minThreshold < 0) { minThreshold = 0;
minSlider.setValue((int)minThreshold);
}
scaleUpAndSet(ip, minThreshold, maxThreshold);
IJ.setKeyUp(KeyEvent.VK_ALT);
IJ.setKeyUp(KeyEvent.VK_SHIFT);
}
void reset(ImagePlus imp, ImageProcessor ip) {
ip.resetThreshold();
ImageStatistics stats = plot.setHistogram(imp, entireStack(imp));
if (!(ip instanceof ByteProcessor)) {
if (entireStack(imp))
ip.setMinAndMax(stats.min, stats.max);
else
ip.resetMinAndMax();
}
updateScrollBars();
if (Recorder.record) {
if (Recorder.scriptMode())
Recorder.recordCall("IJ.resetThreshold(imp);");
else
Recorder.record("resetThreshold");
}
}
void doSet(ImagePlus imp, ImageProcessor ip) {
Calibration cal = imp.getCalibration();
int digits = (ip instanceof FloatProcessor)||cal.calibrated()?2:0;
double level1 = Double.parseDouble(minLabel.getText());
double level2 = Double.parseDouble(maxLabel.getText());
level1 = cal.getRawValue(level1);
level2 = cal.getRawValue(level2);
if (level2<level1)
level2 = level1;
double minDisplay = ip.getMin();
double maxDisplay = ip.getMax();
ip.resetMinAndMax();
double minValue = ip.getMin();
double maxValue = ip.getMax();
if (imp.getStackSize()==1) {
if (level1<minValue) level1 = minValue;
if (level2>maxValue) level2 = maxValue;
}
IJ.wait(500);
ip.setThreshold(level1, level2, lutColor);
ip.setSnapshotPixels(null); previousImageID = 0;
setup(imp, false);
if (Recorder.record) {
if (imp.getBitDepth()==32) {
if (Recorder.scriptMode())
Recorder.recordCall("IJ.setThreshold(imp, "+IJ.d2s(ip.getMinThreshold(),4)+", "+IJ.d2s(ip.getMaxThreshold(),4)+");");
else
Recorder.record("setThreshold", ip.getMinThreshold(), ip.getMaxThreshold());
} else {
int min = (int)ip.getMinThreshold();
int max = (int)ip.getMaxThreshold();
if (cal.isSigned16Bit()) {
min = (int)cal.getCValue(level1);
max = (int)cal.getCValue(level2);
if (Recorder.scriptMode())
Recorder.recordCall("IJ.setThreshold(imp, "+min+", "+max+");");
else
Recorder.record("setThreshold", min, max);
}
if (Recorder.scriptMode())
Recorder.recordCall("IJ.setRawThreshold(imp, "+min+", "+max+", null);");
else {
if (cal.calibrated())
Recorder.record("setThreshold", min, max, "raw");
else
Recorder.record("setThreshold", min, max);
}
}
}
}
void changeState(ImagePlus imp, ImageProcessor ip) {
scaleUpAndSet(ip, minThreshold, maxThreshold);
updateScrollBars();
}
void autoThreshold(ImagePlus imp, ImageProcessor ip) {
ip.resetThreshold();
previousImageID = 0;
setup(imp, true);
}
void apply(ImagePlus imp) {
try {
if (imp.getBitDepth()==32) {
GenericDialog gd = new GenericDialog("NaN Backround");
gd.addCheckbox("Set background pixels to NaN", backgroundToNaN);
gd.showDialog();
if (gd.wasCanceled()) {
runThresholdCommand();
return;
}
backgroundToNaN = gd.getNextBoolean();
if (backgroundToNaN) {
Recorder.recordInMacros = true;
IJ.run("NaN Background");
Recorder.recordInMacros = false;
} else
runThresholdCommand();
} else
runThresholdCommand();
} catch (Exception e)
{}
}
void runThresholdCommand() {
Thresholder.setMethod(method);
Thresholder.setBackground(darkBackground.getState()?"Dark":"Light");
if (Recorder.record) {
Recorder.setCommand("Convert to Mask");
(new Thresholder()).run("mask");
Recorder.saveCommand();
} else
(new Thresholder()).run("mask");
}
static final int RESET=0, AUTO=1, HIST=2, APPLY=3, STATE_CHANGE=4, MIN_THRESHOLD=5, MAX_THRESHOLD=6, SET=7;
public void run() {
while (!done) {
synchronized(this) {
try {wait();}
catch(InterruptedException e) {}
}
doUpdate();
}
}
void doUpdate() {
ImagePlus imp;
ImageProcessor ip;
int action;
int min = minValue;
int max = maxValue;
if (doReset) action = RESET;
else if (doAutoAdjust) action = AUTO;
else if (doApplyLut) action = APPLY;
else if (doStateChange) action = STATE_CHANGE;
else if (doSet) action = SET;
else if (minValue>=0) action = MIN_THRESHOLD;
else if (maxValue>=0) action = MAX_THRESHOLD;
else return;
minValue = -1;
maxValue = -1;
doReset = false;
doAutoAdjust = false;
doApplyLut = false;
doStateChange = false;
doSet = false;
imp = WindowManager.getCurrentImage();
if (imp==null) {
IJ.beep();
IJ.showStatus("No image");
return;
}
ip = setup(imp, false);
if (ip==null) {
imp.unlock();
IJ.beep();
if (imp.isComposite())
IJ.showStatus("\"Composite\" mode images cannot be thresholded");
else
IJ.showStatus("RGB images cannot be thresholded");
return;
}
switch (action) {
case RESET: reset(imp, ip); break;
case AUTO: autoThreshold(imp, ip); break;
case APPLY: apply(imp); break;
case STATE_CHANGE: changeState(imp, ip); break;
case SET: doSet(imp, ip); break;
case MIN_THRESHOLD: adjustMinThreshold(imp, ip, min); break;
case MAX_THRESHOLD: adjustMaxThreshold(imp, ip, max); break;
}
updatePlot(ip);
updateLabels(imp, ip);
updatePercentiles(imp, ip);
ip.setLutAnimation(true);
imp.updateAndDraw();
}
public void close() {
super.close();
instance = null;
done = true;
Prefs.saveLocation(LOC_KEY, getLocation());
Prefs.set(MODE_KEY, mode);
Prefs.set(DARK_BACKGROUND, darkBackground.getState());
synchronized(this) {
notify();
}
}
public void windowActivated(WindowEvent e) {
super.windowActivated(e);
plot.requestFocus();
ImagePlus imp = WindowManager.getCurrentImage();
if (!firstActivation && imp!=null)
setup(imp, false);
}
int roiHashCode(Roi roi) {
return roi!=null?roi.getHashCode():0;
}
public static void update() {
if (instance!=null) {
ThresholdAdjuster ta = ((ThresholdAdjuster)instance);
ImagePlus imp = WindowManager.getCurrentImage();
if (imp!=null && ta.previousImageID==imp.getID()) {
if ((imp.getCurrentSlice()!=ta.previousSlice) && ta.entireStack(imp))
return;
ta.previousImageID = 0;
ta.setup(imp, false);
}
}
}
public static String getMethod() {
return method;
}
public static void setMethod(String thresholdingMethod) {
boolean valid = false;
for (int i=0; i<methodNames.length; i++) {
if (methodNames[i].equals(thresholdingMethod)) {
valid = true;
break;
}
}
if (valid) {
method = thresholdingMethod;
if (instance!=null)
instance.methodChoice.select(method);
}
}
public static String getMode() {
return modes[mode];
}
public static void setMode(String tmode) {
if (instance!=null) synchronized (instance) {
ThresholdAdjuster ta = ((ThresholdAdjuster)instance);
if (modes[0].equals(tmode))
mode = 0;
else if (modes[1].equals(tmode))
mode = 1;
else if (modes[2].equals(tmode))
mode = 2;
else
return;
ta.setLutColor(mode);
ta.doStateChange = true;
ta.modeChoice.select(mode);
ta.notify();
}
}
}
class ThresholdPlot extends Canvas implements Measurements, MouseListener {
static final int WIDTH = 256, HEIGHT=48;
int lowerThreshold = -1;
int upperThreshold = 170;
ImageStatistics stats;
int[] histogram;
Color[] hColors;
int hmax;
Image os;
Graphics osg;
int mode;
int originalModeCount;
double stackMin, stackMax;
int imageID2;
boolean entireStack2;
double mean2;
public ThresholdPlot() {
addMouseListener(this);
setSize(WIDTH+2, HEIGHT+2);
}
public Dimension getPreferredSize() {
return new Dimension(WIDTH+2, HEIGHT+2);
}
ImageStatistics setHistogram(ImagePlus imp, boolean entireStack) {
if (IJ.debugMode) IJ.log("ThresholdAdjuster:setHistogram: "+entireStack+" "+entireStack2);
double mean = entireStack?imp.getProcessor().getStats().mean:0.0;
if (entireStack && stats!=null && imp.getID()==imageID2
&& entireStack==entireStack2 && mean==mean2)
return stats;
mean2 = mean;
ImageProcessor ip = imp.getProcessor();
ColorModel cm = ip.getColorModel();
stats = null;
if (entireStack) {
if (imp.isHyperStack()) {
ImageStack stack = ChannelSplitter.getChannel(imp, imp.getChannel());
stats = new StackStatistics(new ImagePlus("", stack));
} else
stats = new StackStatistics(imp);
}
if (!(ip instanceof ByteProcessor)) {
if (entireStack) {
if (imp.getLocalCalibration().isSigned16Bit())
{stats.min += 32768; stats.max += 32768;}
stackMin = stats.min;
stackMax = stats.max;
ip.setMinAndMax(stackMin, stackMax);
imp.updateAndDraw();
} else {
stackMin = stackMax = 0.0;
if (entireStack2) {
ip.resetMinAndMax();
imp.updateAndDraw();
}
}
Calibration cal = imp.getCalibration();
if (ip instanceof FloatProcessor) {
int digits = Math.max(Analyzer.getPrecision(), 2);
IJ.showStatus("min="+IJ.d2s(ip.getMin(),digits)+", max="+IJ.d2s(ip.getMax(),digits));
} else
IJ.showStatus("min="+(int)cal.getCValue(ip.getMin())+", max="+(int)cal.getCValue(ip.getMax()));
ip = ip.convertToByte(true);
ip.setColorModel(ip.getDefaultColorModel());
}
Roi roi = imp.getRoi();
if (roi!=null && !roi.isArea()) roi = null;
ip.setRoi(roi);
if (stats==null)
stats = ip.getStats();
if (IJ.debugMode) IJ.log(" stats: "+stats);
int maxCount2 = 0;
histogram = stats.histogram;
originalModeCount = histogram[stats.mode];
for (int i = 0; i < stats.nBins; i++)
if ((histogram[i] > maxCount2) && (i != stats.mode))
maxCount2 = histogram[i];
hmax = stats.maxCount;
if ((hmax>(maxCount2 * 2)) && (maxCount2 != 0))
hmax = (int)(maxCount2 * 1.5);
os = null;
if (!(cm instanceof IndexColorModel))
return null;
IndexColorModel icm = (IndexColorModel)cm;
int mapSize = icm.getMapSize();
if (mapSize!=256)
return null;
byte[] r = new byte[256];
byte[] g = new byte[256];
byte[] b = new byte[256];
icm.getReds(r);
icm.getGreens(g);
icm.getBlues(b);
hColors = new Color[256];
final int brightnessLimit = 1800; for (int i=0; i<256; i++) { int sum = 4*(r[i]&255) + 5*(g[i]&255) + (b[i]&255);
if (sum > brightnessLimit) {
r[i] = (byte)(((r[i]&255)*brightnessLimit*2)/(sum+brightnessLimit));
g[i] = (byte)(((g[i]&255)*brightnessLimit*2)/(sum+brightnessLimit));
b[i] = (byte)(((b[i]&255)*brightnessLimit*2)/(sum+brightnessLimit));
}
hColors[i] = new Color(r[i]&255, g[i]&255, b[i]&255);
}
imageID2 = imp.getID();
entireStack2 = entireStack;
return stats;
}
public void update(Graphics g) {
paint(g);
}
public void paint(Graphics g) {
if (g==null) return;
if (histogram!=null) {
if (os==null && hmax>0) {
os = createImage(WIDTH,HEIGHT);
osg = os.getGraphics();
osg.setColor(Color.white);
osg.fillRect(0, 0, WIDTH, HEIGHT);
osg.setColor(Color.gray);
for (int i = 0; i < WIDTH; i++) {
if (hColors!=null) osg.setColor(hColors[i]);
int histValue = histogram[i]<hmax ? histogram[i] : hmax;
osg.drawLine(i, HEIGHT, i, HEIGHT - (HEIGHT*histogram[i]+hmax-1)/hmax);
}
osg.dispose();
}
if (os==null) return;
g.drawImage(os, 1, 1, this);
} else {
g.setColor(Color.white);
g.fillRect(1, 1, WIDTH, HEIGHT);
}
g.setColor(Color.black);
g.drawRect(0, 0, WIDTH+1, HEIGHT+1);
if (lowerThreshold==-1)
return;
if (mode==ThresholdAdjuster.OVER_UNDER) {
g.setColor(Color.blue);
g.drawRect(0, 0, lowerThreshold, HEIGHT+1);
g.drawRect(0, 1, lowerThreshold, 1);
g.setColor(Color.green);
g.drawRect(upperThreshold+2, 0, WIDTH-upperThreshold-1, HEIGHT+1);
g.drawLine(upperThreshold+2, 1, WIDTH+1,1);
return;
}
if (mode==ThresholdAdjuster.RED)
g.setColor(Color.red);
g.drawRect(lowerThreshold+1, 0, upperThreshold-lowerThreshold, HEIGHT+1);
g.drawLine(lowerThreshold+1, 1, upperThreshold+1, 1);
}
public void mousePressed(MouseEvent e) {}
public void mouseReleased(MouseEvent e) {}
public void mouseExited(MouseEvent e) {}
public void mouseClicked(MouseEvent e) {}
public void mouseEntered(MouseEvent e) {}
}