diff --git a/src/main/java/fiji/plugin/trackmate/action/LabelImgExporter.java b/src/main/java/fiji/plugin/trackmate/action/LabelImgExporter.java index 12b40b5fb..7013db5d9 100644 --- a/src/main/java/fiji/plugin/trackmate/action/LabelImgExporter.java +++ b/src/main/java/fiji/plugin/trackmate/action/LabelImgExporter.java @@ -38,6 +38,7 @@ import fiji.plugin.trackmate.SelectionModel; import fiji.plugin.trackmate.Spot; import fiji.plugin.trackmate.TrackMate; +import fiji.plugin.trackmate.TrackModel; import fiji.plugin.trackmate.gui.displaysettings.DisplaySettings; import fiji.plugin.trackmate.util.TMUtils; import fiji.plugin.trackmate.visualization.GlasbeyLut; @@ -51,6 +52,7 @@ import net.imglib2.img.Img; import net.imglib2.img.display.imagej.ImageJFunctions; import net.imglib2.type.numeric.RealType; +import net.imglib2.type.numeric.integer.UnsignedShortType; import net.imglib2.type.numeric.real.FloatType; import net.imglib2.util.Util; import net.imglib2.view.Views; @@ -61,12 +63,11 @@ public class LabelImgExporter extends AbstractTMAction public static final String INFO_TEXT = "" + "This action creates a label image from the tracking results. " + "

" - + "A new 16-bit image is generated, of same dimension and size that " - + "of the input image. The label image has one channel, with black baground (0 value) " + + "A new 32-bit image is generated, of same dimension and size that " + + "of the input image. The label image has one channel, with black background (0 value) " + "everywhere, except where there are spots. Each spot is painted with " - + "a uniform integer value equal to the trackID it belongs to. " - + "Spots that do not belong to tracks are painted with a unique integer " - + "larger than the last trackID in the dataset. " + + "a uniform integer value, configurable to be the spot ID, the ID of the " + + "track it belongs to, or a simple index, unique to the frame or to the movie. " + "

" + "Only visible spots are painted. " + ""; @@ -84,7 +85,7 @@ public void execute( final TrackMate trackmate, final SelectionModel selectionMo final boolean exportSpotsAsDots; final boolean exportTracksOnly; - final boolean useSpotIDsAsLabels; + final LabelIdPainting labelIdPainting; if ( gui != null ) { final LabelImgExporterPanel panel = new LabelImgExporterPanel(); @@ -101,20 +102,20 @@ public void execute( final TrackMate trackmate, final SelectionModel selectionMo exportSpotsAsDots = panel.isExportSpotsAsDots(); exportTracksOnly = panel.isExportTracksOnly(); - useSpotIDsAsLabels = panel.isUseSpotIDsAsLabels();; + labelIdPainting = panel.labelIdPainting(); } else { exportSpotsAsDots = false; exportTracksOnly = false; - useSpotIDsAsLabels = false; + labelIdPainting = LabelIdPainting.LABEL_IS_INDEX_MOVIE_UNIQUE; } /* * Generate label image. */ - createLabelImagePlus( trackmate, exportSpotsAsDots, exportTracksOnly, useSpotIDsAsLabels ,logger ).show(); + createLabelImagePlus( trackmate, exportSpotsAsDots, exportTracksOnly, labelIdPainting, logger ).show(); } /** @@ -130,17 +131,15 @@ public void execute( final TrackMate trackmate, final SelectionModel selectionMo * of this input image, except for the number of channels, which * will be 1. * @param exportSpotsAsDots - * if true, spots will be painted as single dots - * instead of ellipsoids. + * if true, spots will be painted as single dots. If + * false they will be painted with their shape. * @param exportTracksOnly * if true, only the spots belonging to visible * tracks will be painted. If false, spots not * belonging to a track will be painted with a unique ID, * different from the track IDs and different for each spot. - * @param useSpotIDsAsLabels - * if true, the label mask images will contain - * the spot ID. If false, the label mask images - * will contain the track ID. + * @param labeIdPainting + * specifies how to paint the label ID of spots. * * @return a new {@link ImagePlus}. */ @@ -148,10 +147,10 @@ public static final ImagePlus createLabelImagePlus( final TrackMate trackmate, final boolean exportSpotsAsDots, final boolean exportTracksOnly, - final boolean useSpotIDsAsLabels + final LabelIdPainting labeIdPainting ) { - return createLabelImagePlus( trackmate, exportSpotsAsDots, exportTracksOnly, useSpotIDsAsLabels, Logger.VOID_LOGGER ); + return createLabelImagePlus( trackmate, exportSpotsAsDots, exportTracksOnly, labeIdPainting, Logger.VOID_LOGGER ); } /** @@ -174,10 +173,8 @@ public static final ImagePlus createLabelImagePlus( * tracks will be painted. If false, spots not * belonging to a track will be painted with a unique ID, * different from the track IDs and different for each spot. - * @param useSpotIDsAsLabels - * if true, the label mask images will contain - * the spot ID. If false, the label mask images - * will contain the track ID. + * @param labelIdPainting + * specifies how to paint the label ID of spots. * @param logger * a {@link Logger} instance, to report progress of the export * process. @@ -188,10 +185,10 @@ public static final ImagePlus createLabelImagePlus( final TrackMate trackmate, final boolean exportSpotsAsDots, final boolean exportTracksOnly, - final boolean useSpotIDsAsLabels, + final LabelIdPainting labelIdPainting, final Logger logger ) { - return createLabelImagePlus( trackmate.getModel(), trackmate.getSettings().imp, exportSpotsAsDots, exportTracksOnly, useSpotIDsAsLabels, logger ); + return createLabelImagePlus( trackmate.getModel(), trackmate.getSettings().imp, exportSpotsAsDots, exportTracksOnly, labelIdPainting, logger ); } /** @@ -214,10 +211,8 @@ public static final ImagePlus createLabelImagePlus( * tracks will be painted. If false, spots not * belonging to a track will be painted with a unique ID, * different from the track IDs and different for each spot. - * @param useSpotIDsAsLabels - * if true, the label mask images will contain - * the spot ID. If false, the label mask images - * will contain the track ID. + * @param labelIdPainting + * specifies how to paint the label ID of spots. * * @return a new {@link ImagePlus}. */ @@ -226,9 +221,9 @@ public static final ImagePlus createLabelImagePlus( final ImagePlus imp, final boolean exportSpotsAsDots, final boolean exportTracksOnly, - final boolean useSpotIDsAsLabels ) + final LabelIdPainting labelIdPainting ) { - return createLabelImagePlus( model, imp, exportSpotsAsDots, exportTracksOnly, useSpotIDsAsLabels, Logger.VOID_LOGGER ); + return createLabelImagePlus( model, imp, exportSpotsAsDots, exportTracksOnly, labelIdPainting, Logger.VOID_LOGGER ); } /** @@ -251,10 +246,8 @@ public static final ImagePlus createLabelImagePlus( * tracks will be painted. If false, spots not * belonging to a track will be painted with a unique ID, * different from the track IDs and different for each spot. - * @param useSpotIDsAsLabels - * if true, the label mask images will contain - * the spot ID. If false, the label mask images - * will contain the track ID. + * @param labelIdPainting + * specifies how to paint the label ID of spots. * @param logger * a {@link Logger} instance, to report progress of the export * process. @@ -266,7 +259,7 @@ public static final ImagePlus createLabelImagePlus( final ImagePlus imp, final boolean exportSpotsAsDots, final boolean exportTracksOnly, - final boolean useSpotIDsAsLabels, + final LabelIdPainting labelIdPainting, final Logger logger ) { final int[] dimensions = imp.getDimensions(); @@ -277,7 +270,7 @@ public static final ImagePlus createLabelImagePlus( imp.getCalibration().pixelDepth, imp.getCalibration().frameInterval }; - final ImagePlus lblImp = createLabelImagePlus( model, dims, calibration, exportSpotsAsDots, exportTracksOnly, useSpotIDsAsLabels, logger ); + final ImagePlus lblImp = createLabelImagePlus( model, dims, calibration, exportSpotsAsDots, exportTracksOnly, labelIdPainting, logger ); lblImp.setCalibration( imp.getCalibration().copy() ); lblImp.setTitle( "LblImg_" + imp.getTitle() ); return lblImp; @@ -306,10 +299,8 @@ public static final ImagePlus createLabelImagePlus( * tracks will be painted. If false, spots not * belonging to a track will be painted with a unique ID, * different from the track IDs and different for each spot. - * @param useSpotIDsAsLabels - * if true, the label mask images will contain the - * spot ID. If false, the label mask images will - * contain the track ID. + * @param labelIdPainting + * specifies how to paint the label ID of spots. * * @return a new {@link ImagePlus}. */ @@ -319,9 +310,9 @@ public static final ImagePlus createLabelImagePlus( final double[] calibration, final boolean exportSpotsAsDots, final boolean exportTracksOnly, - final boolean useSpotIDsAsLabels) + final LabelIdPainting labelIdPainting ) { - return createLabelImagePlus( model, dimensions, calibration, exportSpotsAsDots, exportTracksOnly, useSpotIDsAsLabels, Logger.VOID_LOGGER ); + return createLabelImagePlus( model, dimensions, calibration, exportSpotsAsDots, exportTracksOnly, labelIdPainting, Logger.VOID_LOGGER ); } /** @@ -346,10 +337,11 @@ public static final ImagePlus createLabelImagePlus( * tracks will be painted. If false, spots not * belonging to a track will be painted with a unique ID, * different from the track IDs and different for each spot. - * @param useSpotIDsAsLabels * if true, the label mask images will contain the * spot ID. If false, the label mask images will * contain the track ID. + * @param labelIdPainting + * specifies how to paint the label ID of spots. * @param logger * a {@link Logger} instance, to report progress of the export * process. @@ -362,14 +354,14 @@ public static final ImagePlus createLabelImagePlus( final double[] calibration, final boolean exportSpotsAsDots, final boolean exportTracksOnly, - final boolean useSpotIDsAsLabels, + final LabelIdPainting labelIdPainting, final Logger logger ) { final long[] dims = new long[ 4 ]; for ( int d = 0; d < dims.length; d++ ) dims[ d ] = dimensions[ d ]; - final ImagePlus lblImp = ImageJFunctions.wrap( createLabelImg( model, dims, calibration, exportSpotsAsDots, exportTracksOnly, useSpotIDsAsLabels, logger ), "LblImage" ); + final ImagePlus lblImp = ImageJFunctions.wrap( createLabelImg( model, dims, calibration, exportSpotsAsDots, exportTracksOnly, labelIdPainting, logger ), "LblImage" ); lblImp.setDimensions( 1, dimensions[ 2 ], dimensions[ 3 ] ); lblImp.setLut( GlasbeyLut.toLUT() ); lblImp.setDisplayRange( 0, 255 ); @@ -399,10 +391,8 @@ public static final ImagePlus createLabelImagePlus( * tracks will be painted. If false, spots not * belonging to a track will be painted with a unique ID, * different from the track IDs and different for each spot. - * @param useSpotIDsAsLabels - * if true, the label mask images will contain the - * spot ID. If false, the label mask images will - * contain the track ID. + * @param labelIdPainting + * specifies how to paint the label ID of spots. * * @return a new {@link Img}. */ @@ -412,9 +402,9 @@ public static final Img< FloatType > createLabelImg( final double[] calibration, final boolean exportSpotsAsDots, final boolean exportTracksOnly, - final boolean useSpotIDsAsLabels) + final LabelIdPainting labelIdPainting ) { - return createLabelImg( model, dimensions, calibration, exportSpotsAsDots, exportTracksOnly, useSpotIDsAsLabels, Logger.VOID_LOGGER ); + return createLabelImg( model, dimensions, calibration, exportSpotsAsDots, exportTracksOnly, labelIdPainting, Logger.VOID_LOGGER ); } /** @@ -439,10 +429,8 @@ public static final Img< FloatType > createLabelImg( * tracks will be painted. If false, spots not * belonging to a track will be painted with a unique ID, * different from the track IDs and different for each spot. - * @param useSpotIDsAsLabels - * if true, the label mask images will contain the - * spot ID. If false, the label mask images will - * contain the track ID. + * @param labelIdPainting + * specifies how to paint the label ID of spots. * @param logger * a {@link Logger} instance, to report progress of the export * process. @@ -455,7 +443,7 @@ public static final Img< FloatType > createLabelImg( final double[] calibration, final boolean exportSpotsAsDots, final boolean exportTracksOnly, - final boolean useSpotIDsAsLabels, + final LabelIdPainting labelIdPainting, final Logger logger ) { /* @@ -471,16 +459,10 @@ public static final Img< FloatType > createLabelImg( final ImgPlus< FloatType > imgPlus = new ImgPlus<>( lblImg, "LblImg", axes, calibration ); /* - * Determine the starting id for spots not in tracks. + * How to assign an ID to spots. */ - int maxTrackID = -1; - final Set< Integer > trackIDs = model.getTrackModel().trackIDs( false ); - if ( null != trackIDs ) - for ( final Integer trackID : trackIDs ) - if ( trackID > maxTrackID ) - maxTrackID = trackID.intValue(); - final AtomicInteger lonelySpotID = new AtomicInteger( maxTrackID + 2 ); + final IdGenerator idGenerator = labelIdPainting.idGenerator( model.getTrackModel(), exportTracksOnly ); /* * Frame by frame iteration. @@ -493,29 +475,11 @@ public static final Img< FloatType > createLabelImg( final SpotWriter spotWriter = exportSpotsAsDots ? new SpotAsDotWriter<>( imgCT ) : new SpotShapeWriter<>( imgCT ); + idGenerator.nextFrame(); for ( final Spot spot : model.getSpots().iterable( frame, true ) ) { - final int id; - final Integer trackID = model.getTrackModel().trackIDOf( spot ); - if ( null == trackID || !model.getTrackModel().isVisible( trackID ) ) - { - if ( exportTracksOnly ) - continue; - - if ( useSpotIDsAsLabels ) - id = spot.ID() + 1; - else - id = lonelySpotID.getAndIncrement(); - } - else - { - if ( useSpotIDsAsLabels ) - id = spot.ID() + 1; - else - id = 1 + trackID.intValue(); - } - + final int id = idGenerator.id( spot ); spotWriter.write( spot, id ); } logger.setProgress( ( double ) ( 1 + frame ) / dimensions[ 3 ] ); @@ -612,4 +576,176 @@ public void write( final Spot spot, final int id ) ra.get().setReal( id ); } } + + /** + * Specifies how spots will be labeled in the exported label image. + */ + public static enum LabelIdPainting + { + + LABEL_IS_TRACK_ID( "Track ID", + "The spot label is the ID of the track it belongs to, plus one (+1). " + + "Spots that do not belong to tracks are painted with a unique integer " + + "larger than the last trackID in the dataset." ), + LABEL_IS_SPOT_ID( "Spot ID", + "The spot label is the spot ID, plus one (+1)." ), + LABEL_IS_INDEX( "Index unique in frame", + "The spot label is an index starting from 1. " + + "It is unique within a frame, " + + "but can be repeated across frames." ), + LABEL_IS_INDEX_MOVIE_UNIQUE( "Unique index", + "The spot label is an index starting from 1. " + + "It is unique within the whole movie." ); + + private final String info; + + private final String methodName; + + private LabelIdPainting( final String methodName, final String info ) + { + this.methodName = methodName; + this.info = info; + } + + @Override + public String toString() + { + return methodName; + } + + public String getInfo() + { + return info; + } + + public IdGenerator idGenerator( final TrackModel tm, final boolean visibleTracksOnly) + { + switch ( this ) + { + case LABEL_IS_INDEX: + return new SpotIndexGeneratorUniqueInFrame( tm, visibleTracksOnly ); + case LABEL_IS_INDEX_MOVIE_UNIQUE: + return new SpotIndexGeneratorUniqueInMovie( tm, visibleTracksOnly ); + case LABEL_IS_SPOT_ID: + return new SpotIdGenerator( tm, visibleTracksOnly ); + case LABEL_IS_TRACK_ID: + return new TrackIdGenerator( tm, visibleTracksOnly ); + default: + throw new IllegalArgumentException( "Unknown painting id mode: " + this ); + } + } + } + + private static interface IdGenerator + { + public int id( Spot spot ); + + public default void nextFrame() + {}; + } + + private static abstract class AbstractIdGenerator implements IdGenerator + { + protected final TrackModel tm; + + protected final boolean visibleTracksOnly; + + public AbstractIdGenerator( final TrackModel tm, final boolean visibleTracksOnly ) + { + this.tm = tm; + this.visibleTracksOnly = visibleTracksOnly; + } + } + + private static class TrackIdGenerator extends AbstractIdGenerator + { + + private final AtomicInteger lonelySpotID; + + public TrackIdGenerator( final TrackModel tm, final boolean visibleTracksOnly ) + { + super( tm, visibleTracksOnly ); + int maxTrackID = -1; + final Set< Integer > trackIDs = tm.trackIDs( false ); + if ( null != trackIDs ) + for ( final Integer trackID : trackIDs ) + if ( trackID > maxTrackID ) + maxTrackID = trackID.intValue(); + this.lonelySpotID = new AtomicInteger( maxTrackID + 2 ); + } + + @Override + public int id( final Spot spot ) + { + final Integer trackID = tm.trackIDOf( spot ); + if ( null == trackID || !tm.isVisible( trackID ) ) + { + if ( visibleTracksOnly ) + return -1; + return lonelySpotID.getAndIncrement(); + } + return trackID + 1; + } + } + + private static class SpotIdGenerator extends AbstractIdGenerator + { + + public SpotIdGenerator( final TrackModel tm, final boolean visibleTracksOnly ) + { + super( tm, visibleTracksOnly ); + } + + @Override + public int id( final Spot spot ) + { + final Integer trackID = tm.trackIDOf( spot ); + if ( null == trackID || !tm.isVisible( trackID ) ) + { + if ( visibleTracksOnly ) + return -1; + } + return spot.ID() + 1; + } + } + + private static class SpotIndexGeneratorUniqueInMovie extends AbstractIdGenerator + { + + protected final AtomicInteger index; + + public SpotIndexGeneratorUniqueInMovie( final TrackModel tm, final boolean visibleTracksOnly ) + { + super( tm, visibleTracksOnly ); + this.index = new AtomicInteger( 0 ); + } + + @Override + public int id( final Spot spot ) + { + final Integer trackID = tm.trackIDOf( spot ); + if ( null == trackID || !tm.isVisible( trackID ) ) + { + if ( visibleTracksOnly ) + return -1; + } + return index.incrementAndGet(); + } + } + + + private static class SpotIndexGeneratorUniqueInFrame extends SpotIndexGeneratorUniqueInMovie + { + + public SpotIndexGeneratorUniqueInFrame( final TrackModel tm, final boolean visibleTracksOnly ) + { + super( tm, visibleTracksOnly ); + } + + @Override + public void nextFrame() + { + index.set( 0 ); + } + } } diff --git a/src/main/java/fiji/plugin/trackmate/action/LabelImgExporterPanel.java b/src/main/java/fiji/plugin/trackmate/action/LabelImgExporterPanel.java index 9aebd3688..31366fd60 100644 --- a/src/main/java/fiji/plugin/trackmate/action/LabelImgExporterPanel.java +++ b/src/main/java/fiji/plugin/trackmate/action/LabelImgExporterPanel.java @@ -21,13 +21,19 @@ */ package fiji.plugin.trackmate.action; +import java.awt.Dimension; import java.awt.GridBagConstraints; import java.awt.GridBagLayout; import java.awt.Insets; import javax.swing.JCheckBox; +import javax.swing.JComboBox; +import javax.swing.JLabel; import javax.swing.JPanel; +import fiji.plugin.trackmate.action.LabelImgExporter.LabelIdPainting; +import fiji.plugin.trackmate.gui.Fonts; + public class LabelImgExporterPanel extends JPanel { @@ -37,15 +43,17 @@ public class LabelImgExporterPanel extends JPanel private final JCheckBox exportTracksOnly; - private final JCheckBox useSpotIdsAsLabels; + private final JComboBox< LabelIdPainting > labelIdPainting; public LabelImgExporterPanel() { + setPreferredSize( new Dimension( 250, 200 ) ); final GridBagLayout gridBagLayout = new GridBagLayout(); + gridBagLayout.rowWeights = new double[] { 0., 0., 0., 1. }; setLayout( gridBagLayout ); final GridBagConstraints gbc = new GridBagConstraints(); - gbc.anchor = GridBagConstraints.WEST; + gbc.anchor = GridBagConstraints.NORTHWEST; gbc.insets = new Insets( 5, 5, 5, 5 ); gbc.gridx = 0; gbc.gridy = 0; @@ -57,9 +65,24 @@ public LabelImgExporterPanel() gbc.gridy++; add( exportTracksOnly, gbc ); - useSpotIdsAsLabels = new JCheckBox( "Use spots IDs as labels", false ); gbc.gridy++; - add( useSpotIdsAsLabels, gbc ); + add( new JLabel( "Label ID is:" ), gbc ); + + labelIdPainting = new JComboBox<>( LabelIdPainting.values() ); + gbc.gridy++; + add( labelIdPainting, gbc ); + + final JLabel info = new JLabel(); + info.setFont( Fonts.SMALL_FONT ); + gbc.gridy++; + gbc.fill = GridBagConstraints.BOTH; + add( info, gbc ); + + labelIdPainting.addItemListener( e -> info.setText( "" + + ( ( LabelIdPainting ) labelIdPainting.getSelectedItem() ).getInfo() + + "" ) ); + labelIdPainting.setSelectedIndex( 1 ); + labelIdPainting.setSelectedIndex( 0 ); } public boolean isExportSpotsAsDots() @@ -72,8 +95,8 @@ public boolean isExportTracksOnly() return exportTracksOnly.isSelected(); } - public boolean isUseSpotIDsAsLabels() + public LabelIdPainting labelIdPainting() { - return useSpotIdsAsLabels.isSelected(); + return ( LabelIdPainting ) labelIdPainting.getSelectedItem(); } }