diff --git a/src/main/java/bdv/fx/viewer/render/MultiResolutionRendererFX.java b/src/main/java/bdv/fx/viewer/render/MultiResolutionRendererFX.java index 590b555d2..15b44bb97 100644 --- a/src/main/java/bdv/fx/viewer/render/MultiResolutionRendererFX.java +++ b/src/main/java/bdv/fx/viewer/render/MultiResolutionRendererFX.java @@ -36,9 +36,12 @@ import net.imglib2.Interval; import net.imglib2.Volatile; import net.imglib2.img.array.ArrayImg; +import net.imglib2.img.basictypeaccess.IntAccess; +import net.imglib2.realtransform.AffineTransform3D; import net.imglib2.type.numeric.ARGBType; import net.imglib2.ui.RenderTarget; import net.imglib2.ui.Renderer; +import net.imglib2.ui.TransformListener; /** * A {@link Renderer} that uses a coarse-to-fine rendering scheme. First, a small {@link ArrayImg} at a fraction of @@ -88,15 +91,15 @@ public class MultiResolutionRendererFX extends MultiResolutionRendererGeneric + implements RenderOutputImage.Factory { @Override - public BufferExposingWritableImage create(final int width, final int height) + public RenderOutputImage create(final int width, final int height) { try { - return new BufferExposingWritableImage(width, height); + return new MyRenderOutputImage(new BufferExposingWritableImage(width, height)); } catch (final Exception e) { throw e instanceof RuntimeException ? (RuntimeException) e : new RuntimeException(e); @@ -104,13 +107,38 @@ public BufferExposingWritableImage create(final int width, final int height) } @Override - public BufferExposingWritableImage create(final int width, final int height, final BufferExposingWritableImage - other) - { - // TODO can we somehow re-use smaller image? + public RenderOutputImage create(int width, int height, RenderOutputImage other) { return create(width, height); } + } + + public static class MyRenderOutputImage implements RenderOutputImage { + + private final BufferExposingWritableImage image; + + public MyRenderOutputImage(BufferExposingWritableImage image) { + this.image = image; + } + + @Override + public int width() { + return (int) image.getWidth(); + } + @Override + public int height() { + return (int) image.getHeight(); + } + + @Override + public ArrayImg asArrayImg() { + return image.asArrayImg(); + } + + @Override + public BufferExposingWritableImage unwrap() { + return image; + } } public MultiResolutionRendererFX( @@ -136,11 +164,7 @@ public MultiResolutionRendererFX( useVolatileIfAvailable, accumulateProjectorFactory, cacheControl, - BufferExposingWritableImage::asArrayImg, - new MakeWritableImage(), - img -> (int) img.getWidth(), - img -> (int) img.getHeight() + new MakeWritableImage() ); } - } diff --git a/src/main/java/bdv/fx/viewer/render/MultiResolutionRendererGeneric.java b/src/main/java/bdv/fx/viewer/render/MultiResolutionRendererGeneric.java index 033b27bda..b7f2a61d9 100644 --- a/src/main/java/bdv/fx/viewer/render/MultiResolutionRendererGeneric.java +++ b/src/main/java/bdv/fx/viewer/render/MultiResolutionRendererGeneric.java @@ -61,7 +61,6 @@ import net.imglib2.converter.Converter; import net.imglib2.img.array.ArrayImg; import net.imglib2.img.array.ArrayImgs; -import net.imglib2.img.basictypeaccess.IntAccess; import net.imglib2.img.basictypeaccess.array.IntArray; import net.imglib2.realtransform.AffineTransform3D; import net.imglib2.realtransform.RealViews; @@ -85,7 +84,8 @@ import java.util.Optional; import java.util.concurrent.ExecutorService; import java.util.function.Function; -import java.util.function.ToIntFunction; +import java.util.stream.Collectors; +import java.util.stream.DoubleStream; /** * @@ -98,15 +98,6 @@ public class MultiResolutionRendererGeneric private static final Logger LOG = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass()); - public interface ImageGenerator - { - - T create(int width, int height); - - T create(int width, int height, T other); - - } - /** * Receiver for the data store that we render. */ @@ -118,7 +109,7 @@ public interface ImageGenerator private final PainterThread painterThread; /** - * Currently active projector, used to re-paint the display. It maps the source data to {@link #screenImages}. + * Currently active projector, used to re-paint the display. It maps the source data to {@link ScreenScale#screenImages}. */ private VolatileProjector projector; @@ -133,7 +124,7 @@ public interface ImageGenerator private final boolean doubleBuffered; /** - * Double-buffer index of next {@link #screenImages image} to render. + * Double-buffer index of next {@link ScreenScale#screenImages image} to render. */ private final ArrayDeque renderIdQueue; @@ -143,47 +134,70 @@ public interface ImageGenerator private final HashMap bufferedImageToRenderId; /** - * Used to render an individual source. One image per screen resolution and visible source. First index is screen - * scale, second index is index in list of visible sources. - */ - ArrayImg[][] renderImages; - - /** - * Storage for mask images of {@link VolatileHierarchyProjector}. One array per visible source. (First) index is - * index in list of visible sources. + * Storage for mask images of {@link VolatileHierarchyProjector}. + * One array per visible source. (First) index is index in list of visible sources. */ private byte[][] renderMaskArrays; /** - * Used to render the image for display. Three images per screen resolution if double buffering is enabled. First - * index is screen scale, second index is double-buffer. + * List of scale factors and associate image buffers */ - private List> screenImages; + private List< ScreenScale< T > > screenScales; /** - * data store wrapping the data in the {@link #screenImages}. First index is screen scale, second index is - * double-buffer. + * Scale factor and associated image buffers and transformation. */ - private List> bufferedImages; + private static class ScreenScale< T > { + + /** + * Scale factors from the {@link #display viewer canvas} to the + * {@link #screenImages}. + * + * A scale factor of 1 means 1 pixel in the screen image is displayed as 1 + * pixel on the canvas, a scale factor of 0.5 means 1 pixel in the screen + * image is displayed as 2 pixel on the canvas, etc. + */ + private double scaleFactor; - /** - * Scale factors from the {@link #display viewer canvas} to the {@link #screenImages}. - *

- * A scale factor of 1 means 1 pixel in the screen image is displayed as 1 pixel on the canvas, a scale factor of - * 0.5 means 1 pixel in the screen image is displayed as 2 pixel on the canvas, etc. - */ - private double[] screenScales; + /** + * Used to render an individual source. One image per visible source + * Index is index in list of visible sources. + */ + // this + private List< ArrayImg > renderImages = new ArrayList<>(); + + /** + * Used to render the image for display. Three images if double buffering is + * enabled. + */ + // this + private List< RenderOutputImage< T > > screenImages = new ArrayList<>( Collections.nCopies( 3, null ) ); + + /** + * {@link RenderOutputImage< T >}s wrapping the data in the {@link #screenImages}. + * Three images if double buffering is enabled. + */ + // this + private List< RenderOutputImage< T > > bufferedImages = new ArrayList<>( Collections.nCopies( 3, null ) ); + + /** + * The scale transformation from viewer to {@link #screenImages screen + * image}. + */ + // this + private AffineTransform3D screenScaleTransforms = new AffineTransform3D(); + + /** + * Pending repaint requests. + */ + private Interval pendingRepaintRequests; + + private ScreenScale(double scaleFactor) { + this.scaleFactor = scaleFactor; + } + } - /** - * The scale transformation from viewer to {@link #screenImages screen image}. Each transformation corresponds - * to a {@link #screenScales screen scale}. - */ - private AffineTransform3D[] screenScaleTransforms; - /** - * Pending repaint requests for each {@link #screenScales screen scale}. - */ - private Interval[] pendingRepaintRequests; /** * The last rendered interval in screen space. @@ -248,13 +262,14 @@ public interface ImageGenerator private final boolean useVolatileIfAvailable; /** - * Whether a repaint was {@link #requestRepaint(Interval) requested}. This will cause {@link - * CacheControl#prepareNextFrame()}. + * Whether a repaint was {@link #requestRepaint(Interval) requested}. This will + * cause {@link CacheControl#prepareNextFrame()}. */ private boolean newFrameRequest; /** - * The timepoint for which last a projector was {@link #createProjector created}. + * The timepoint for which last a projector was + * {@link #createProjector created}. */ private int previousTimepoint; @@ -262,13 +277,7 @@ public interface ImageGenerator private boolean prefetchCells = true; - private final Function> wrapAsArrayImg; - - private final ToIntFunction width; - - private final ToIntFunction height; - - private final ImageGenerator makeImage; + private final RenderOutputImage.Factory makeImage; private final AffineTransform3D currentProjectorTransform = new AffineTransform3D(); @@ -276,24 +285,29 @@ public interface ImageGenerator * @param display * The canvas that will display the images we render. * @param painterThread - * Thread that triggers repainting of the display. Requests for repainting are send there. + * Thread that triggers repainting of the display. Requests for + * repainting are send there. * @param screenScales - * Scale factors from the viewer canvas to screen images of different resolutions. A scale factor of 1 means 1 - * pixel in the screen image is displayed as 1 pixel on the canvas, a scale factor of 0.5 means 1 pixel in the - * screen image is displayed as 2 pixel on the canvas, etc. + * Scale factors from the viewer canvas to screen images of + * different resolutions. A scale factor of 1 means 1 pixel in + * the screen image is displayed as 1 pixel on the canvas, a + * scale factor of 0.5 means 1 pixel in the screen image is + * displayed as 2 pixel on the canvas, etc. * @param targetRenderNanos - * Target rendering time in nanoseconds. The rendering time for the coarsest rendered scale should be below - * this - * threshold. + * Target rendering time in nanoseconds. The rendering time for + * the coarsest rendered scale should be below this threshold. * @param doubleBuffered * Whether to use double buffered rendering. * @param numRenderingThreads * How many threads to use for rendering. * @param renderingExecutorService - * if non-null, this is used for rendering. Note, that it is still important to supply the numRenderingThreads - * parameter, because that is used to determine into how many sub-tasks rendering is split. + * if non-null, this is used for rendering. Note, that it is + * still important to supply the numRenderingThreads parameter, + * because that is used to determine into how many sub-tasks + * rendering is split. * @param useVolatileIfAvailable - * whether volatile versions of sources should be used if available. + * whether volatile versions of sources should be used if + * available. * @param accumulateProjectorFactory * can be used to customize how sources are combined. * @param cacheControl @@ -310,29 +324,19 @@ public interface ImageGenerator final boolean useVolatileIfAvailable, final AccumulateProjectorFactory accumulateProjectorFactory, final CacheControl cacheControl, - final Function> wrapAsArrayImg, - final ImageGenerator makeImage, - final ToIntFunction width, - final ToIntFunction height) + final RenderOutputImage.Factory< T > makeImage) { this.display = display; this.painterThread = painterThread; projector = null; currentScreenScaleIndex = -1; - this.screenScales = screenScales.clone(); this.doubleBuffered = doubleBuffered; renderIdQueue = new ArrayDeque<>(); bufferedImageToRenderId = new HashMap<>(); - createVariables(); + createVariables(screenScales); this.makeImage = makeImage; - this.width = width; - - this.height = height; - - this.wrapAsArrayImg = wrapAsArrayImg; - this.targetRenderNanos = targetRenderNanos; renderingMayBeCancelled = true; @@ -346,8 +350,8 @@ public interface ImageGenerator } /** - * Check whether the size of the display component was changed and recreate {@link #screenImages} and {@link - * #screenScaleTransforms} accordingly. + * Check whether the size of the display component was changed and + * recreate {@link ScreenScale#screenImages} and {@link ScreenScale#screenScaleTransforms} accordingly. * * @return whether the size was changed. */ @@ -355,16 +359,16 @@ private synchronized boolean checkResize() { final int componentW = display.getWidth(); final int componentH = display.getHeight(); - if (screenImages.get(0).get(0) == null - || width .applyAsInt(screenImages.get(0).get(0)) != (int) Math.ceil(componentW * screenScales[0]) - || height.applyAsInt(screenImages.get(0).get(0)) != (int) Math.ceil(componentH * screenScales[0])) + if (screenScales.get( 0 ).screenImages.get(0) == null + || screenScales.get( 0 ).screenImages.get(0).width() != (int) Math.ceil(componentW * screenScales.get(0).scaleFactor) + || screenScales.get( 0 ).screenImages.get(0).height() != (int) Math.ceil(componentH * screenScales.get(0).scaleFactor)) { renderIdQueue.clear(); renderIdQueue.addAll(Arrays.asList(0, 1, 2)); bufferedImageToRenderId.clear(); - for (int i = 0; i < screenScales.length; ++i) + for (int i = 0; i < screenScales.size(); ++i) { - final double screenToViewerScale = screenScales[i]; + final double screenToViewerScale = screenScales.get(i).scaleFactor; final int w = (int) Math.ceil(screenToViewerScale * componentW); final int h = (int) Math.ceil(screenToViewerScale * componentH); if (doubleBuffered) @@ -372,19 +376,20 @@ private synchronized boolean checkResize() for (int b = 0; b < 3; ++b) { // reuse storage arrays of level 0 (highest resolution) - screenImages.get(i).set(b, i == 0 - ? makeImage.create(w, h) - : makeImage.create(w, h, screenImages.get(0).get(b))); - final T bi = screenImages.get(i).get(b); + final RenderOutputImage< T > screenImage = ( i == 0 ) ? + makeImage.create( w, h ) : + makeImage.create( w, h, screenScales.get( 0 ).screenImages.get( b ) ); + screenScales.get( i ).screenImages.set( b, screenImage ); + final RenderOutputImage< T > bi = screenScales.get( i ).screenImages.get(b); // getBufferedImage.apply( screenImages[ i ][ b ] ); - bufferedImages.get(i).set(b, bi); - bufferedImageToRenderId.put(bi, b); + screenScales.get( i ).bufferedImages.set(b, bi); + bufferedImageToRenderId.put(bi.unwrap(), b); } } else { - screenImages.get(i).set(0, makeImage.create(w, h)); - bufferedImages.get(i).set(0, screenImages.get(i).get(0)); + screenScales.get( i ).screenImages.set(0, makeImage.create(w, h)); + screenScales.get( i ).bufferedImages.set(0, screenScales.get( i ).screenImages.get(0)); // getBufferedImage.apply( screenImages[ i ][ 0 ] ); } final AffineTransform3D scale = new AffineTransform3D(); @@ -394,7 +399,7 @@ private synchronized boolean checkResize() scale.set(yScale, 1, 1); scale.set(0.5 * (xScale - 1), 0, 3); scale.set(0.5 * (yScale - 1), 1, 3); - screenScaleTransforms[i] = scale; + screenScales.get( i ).screenScaleTransforms = scale; } return true; @@ -406,20 +411,23 @@ private synchronized boolean checkResize() private boolean checkRenewRenderImages(final int numVisibleSources) { final int n = numVisibleSources > 1 ? numVisibleSources : 0; - if (n != renderImages[0].length || + if (n != screenScales.get(0).renderImages.size() || n != 0 && - (renderImages[0][0].dimension(0) != width.applyAsInt(screenImages.get(0).get(0)) || - renderImages[0][0].dimension(1) != height.applyAsInt(screenImages.get(0).get(0)))) + (screenScales.get(0).renderImages.get( 0 ).dimension(0) != screenScales.get( 0 ).screenImages.get(0).width() || + screenScales.get(0).renderImages.get( 0 ).dimension(1) != screenScales.get( 0 ).screenImages.get(0).height())) { - renderImages = new ArrayImg[screenScales.length][n]; - for (int i = 0; i < screenScales.length; ++i) + for (int i = 0; i < screenScales.size(); ++i) { - final int w = width.applyAsInt(screenImages.get(i).get(0)); - final int h = height.applyAsInt(screenImages.get(i).get(0)); + screenScales.get( i ).renderImages = new ArrayList<>( Collections.nCopies( n, null) ); + final int w = ( int ) screenScales.get( i ).screenImages.get( 0 ).width(); + final int h = ( int ) screenScales.get( i ).screenImages.get( 0 ).height(); for (int j = 0; j < n; ++j) - renderImages[i][j] = i == 0 - ? ArrayImgs.argbs(w, h) - : ArrayImgs.argbs(renderImages[0][j].update(null), w, h); + { + final ArrayImg renderImage = (i == 0) + ? ArrayImgs.argbs(w, h) + : ArrayImgs.argbs(screenScales.get(0).renderImages.get(j).update(null), w, h); + screenScales.get( i ).renderImages.set( j, renderImage ); + } } return true; } @@ -428,7 +436,7 @@ private boolean checkRenewRenderImages(final int numVisibleSources) private boolean checkRenewMaskArrays(final int numVisibleSources) { - final int size = width.applyAsInt(screenImages.get(0).get(0)) * height.applyAsInt(screenImages.get(0).get(0)); + final int size = screenScales.get( 0 ).screenImages.get(0).width() * screenScales.get( 0 ).screenImages.get(0).height(); if (numVisibleSources != renderMaskArrays.length || numVisibleSources != 0 && renderMaskArrays[0].length < size) { @@ -440,9 +448,9 @@ private boolean checkRenewMaskArrays(final int numVisibleSources) return false; } - private int[] getImageSize(final T image) + private int[] getImageSize(final RenderOutputImage< T > image) { - return new int[] {this.width.applyAsInt(image), this.height.applyAsInt(image)}; + return new int[] {image.width(), image.height()}; } private static Interval padInterval(final Interval interval, final int[] padding, final int[] imageSize) @@ -472,7 +480,7 @@ public int paint( final boolean resized = checkResize(); // the BufferedImage that is rendered to (to paint to the canvas) - final T bufferedImage; + final RenderOutputImage< T > bufferedImage; // the projector that paints to the screenImage. final VolatileProjector p; @@ -489,11 +497,11 @@ public int paint( // Screen scales are first initialized with the default setting (see RenderUnit), // then the project metadata is loaded, and the screen scales are changed to the saved configuration. // If the project screen scales are [1.0], sometimes the renderer receives a request to re-render the screen at screen scale 1, which results in the exception. - if (requestedScreenScaleIndex >= pendingRepaintRequests.length) + if (requestedScreenScaleIndex >= screenScales.size()) return -1; - repaintScreenInterval = pendingRepaintRequests[requestedScreenScaleIndex]; - pendingRepaintRequests[requestedScreenScaleIndex] = null; + repaintScreenInterval = screenScales.get(requestedScreenScaleIndex).pendingRepaintRequests; + screenScales.get(requestedScreenScaleIndex).pendingRepaintRequests = null; if (repaintScreenInterval == null) return -1; @@ -515,8 +523,8 @@ public int paint( { final int renderId = renderIdQueue.peek(); currentScreenScaleIndex = requestedScreenScaleIndex; - bufferedImage = bufferedImages.get(currentScreenScaleIndex).get(renderId); - final T renderTarget = screenImages.get(currentScreenScaleIndex).get(renderId); + bufferedImage = screenScales.get(currentScreenScaleIndex).bufferedImages.get(renderId); + final RenderOutputImage< T > renderTarget = screenScales.get(currentScreenScaleIndex).screenImages.get(renderId); synchronized (Optional.ofNullable(synchronizationLock).orElse(this)) { final int numSources = sacs.size(); @@ -525,7 +533,7 @@ public int paint( // find the scaling ratio between render target pixels and screen pixels final double[] renderTargetToScreenPixelRatio = new double[2]; - Arrays.setAll(renderTargetToScreenPixelRatio, d -> screenScaleTransforms[currentScreenScaleIndex].get(d, d)); + Arrays.setAll(renderTargetToScreenPixelRatio, d -> screenScales.get(currentScreenScaleIndex).screenScaleTransforms.get(d, d)); // scale the screen repaint request interval into render target coordinates final double[] renderTargetRealIntervalMin = new double[2], renderTargetRealIntervalMax = new double[2]; @@ -546,7 +554,7 @@ public int paint( 0 ); - final RandomAccessibleInterval renderTargetRoi = Views.interval(wrapAsArrayImg.apply(renderTarget), renderTargetPaddedInterval); + final RandomAccessibleInterval renderTargetRoi = Views.interval(renderTarget.asArrayImg(), renderTargetPaddedInterval); p = createProjector( sacs, @@ -583,7 +591,7 @@ public int paint( { if (createProjector) { - final T bi = display.setBufferedImageAndTransform(bufferedImage, currentProjectorTransform); + final T bi = display.setBufferedImageAndTransform(bufferedImage.unwrap(), currentProjectorTransform); if (doubleBuffered) { renderIdQueue.pop(); @@ -610,7 +618,7 @@ public int paint( */ // if (currentScreenScaleIndex == maxScreenScaleIndex) // { -// if (rendertime > targetRenderNanos && maxScreenScaleIndex < screenScales.length - 1) +// if (rendertime > targetRenderNanos && maxScreenScaleIndex < screenScales.size() - 1) // maxScreenScaleIndex++; // else if (rendertime < targetRenderNanos / 3 && maxScreenScaleIndex > 0) // maxScreenScaleIndex--; @@ -638,10 +646,10 @@ else if (!p.isValid()) else { // Add the requested interval back into the queue if it was not rendered - if (pendingRepaintRequests[currentScreenScaleIndex] == null) - pendingRepaintRequests[currentScreenScaleIndex] = repaintScreenInterval; + if (screenScales.get(currentScreenScaleIndex).pendingRepaintRequests == null) + screenScales.get(currentScreenScaleIndex).pendingRepaintRequests = repaintScreenInterval; else - pendingRepaintRequests[currentScreenScaleIndex] = Intervals.union(pendingRepaintRequests[currentScreenScaleIndex], repaintScreenInterval); + screenScales.get(currentScreenScaleIndex).pendingRepaintRequests = Intervals.union(screenScales.get(currentScreenScaleIndex).pendingRepaintRequests, repaintScreenInterval); } return success ? currentScreenScaleIndex : -1; @@ -686,13 +694,13 @@ public synchronized void requestRepaint(final Interval interval, final int scree // Screen scales are first initialized with the default setting (see RenderUnit), // then the project metadata is loaded, and the screen scales are changed to the saved configuration. // If the project screen scales are [1.0], sometimes the renderer receives a request to re-render the screen at screen scale 1, which results in the exception. - if (requestedScreenScaleIndex >= pendingRepaintRequests.length) + if (requestedScreenScaleIndex >= screenScales.size()) return; - if (pendingRepaintRequests[requestedScreenScaleIndex] == null) - pendingRepaintRequests[requestedScreenScaleIndex] = interval; + if (screenScales.get(requestedScreenScaleIndex).pendingRepaintRequests == null) + screenScales.get(requestedScreenScaleIndex).pendingRepaintRequests = interval; else - pendingRepaintRequests[requestedScreenScaleIndex] = Intervals.union(pendingRepaintRequests[requestedScreenScaleIndex], interval); + screenScales.get(requestedScreenScaleIndex).pendingRepaintRequests = Intervals.union(screenScales.get(requestedScreenScaleIndex).pendingRepaintRequests, interval); painterThread.requestRepaint(); } @@ -720,7 +728,7 @@ else if (sacs.size() == 1) LOG.debug("Got only one source, creating pre-multiplying single source projector"); final SourceAndConverter sac = sacs.get(0); final Interpolation interpolation = interpolationForSource.apply(sac.getSpimSource()); - final int[] renderTargetSize = getImageSize(this.screenImages.get(currentScreenScaleIndex).get(0)); + final int[] renderTargetSize = getImageSize(screenScales.get(currentScreenScaleIndex).screenImages.get(0)); projector = createSingleSourceProjector( sac, axisOrders.apply(sac.getSpimSource()), @@ -742,12 +750,12 @@ else if (sacs.size() == 1) int j = 0; for (final SourceAndConverter sac : sacs) { - final RandomAccessibleInterval renderImage = Views.interval(renderImages[currentScreenScaleIndex][j], screenImage); + final RandomAccessibleInterval renderImage = Views.interval(screenScales.get(currentScreenScaleIndex).renderImages.get(j), screenImage); final byte[] maskArray = renderMaskArrays[j]; final AxisOrder axisOrder = axisOrders.apply(sac.getSpimSource()); ++j; final Interpolation interpolation = interpolationForSource.apply(sac.getSpimSource()); - final int[] renderTargetSize = getImageSize(this.screenImages.get(currentScreenScaleIndex).get(0)); + final int[] renderTargetSize = getImageSize(screenScales.get(currentScreenScaleIndex).screenImages.get(0)); final VolatileProjector p = createSingleSourceProjector( sac, axisOrder, @@ -861,7 +869,7 @@ else if (source.getSpimSource().getType() instanceof Volatile) ); } - final AffineTransform3D screenScaleTransform = screenScaleTransforms[currentScreenScaleIndex]; + final AffineTransform3D screenScaleTransform = screenScales.get(currentScreenScaleIndex).screenScaleTransforms; final AffineTransform3D screenTransform = viewerTransform.copy(); screenTransform.preConcatenate(screenScaleTransform); final int bestLevel = MipmapTransforms.getBestMipMapLevel(screenTransform, source.getSpimSource(), timepoint); @@ -897,7 +905,7 @@ private > VolatileProjector createSingleSourceVolatileProj source.getSpimSource(), source.getSpimSource().getName() ); - final AffineTransform3D screenScaleTransform = screenScaleTransforms[currentScreenScaleIndex]; + final AffineTransform3D screenScaleTransform = screenScales.get(currentScreenScaleIndex).screenScaleTransforms; final ArrayList> renderList = new ArrayList<>(); final Source spimSource = source.getSpimSource(); LOG.debug("Creating single source volatile projector for type={}", spimSource.getType()); @@ -1056,8 +1064,7 @@ private static void prefetch( public synchronized void setScreenScales(final double[] screenScales) { - this.screenScales = screenScales.clone(); - createVariables(); + createVariables(screenScales); } /** @@ -1068,27 +1075,18 @@ public synchronized void setScreenScales(final double[] screenScales) */ public synchronized void getScreenScaleTransform(final int screenScaleIndex, final AffineTransform3D screenScaleTransform) { - if (screenScaleIndex < this.screenScaleTransforms.length && this.screenScaleTransforms[screenScaleIndex] != null) - screenScaleTransform.set(this.screenScaleTransforms[screenScaleIndex]); + if (screenScaleIndex < screenScales.size() && screenScales.get(screenScaleIndex).screenScaleTransforms != null) + screenScaleTransform.set(this.screenScales.get(screenScaleIndex).screenScaleTransforms); } - private synchronized void createVariables() + private synchronized void createVariables(double[] screenScales) { - LOG.debug("Updating images for screen scales {}", screenScales); + LOG.debug("Updating images for screen scales {}", this.screenScales); if (renderingMayBeCancelled && projector != null) projector.cancel(); - renderImages = new ArrayImg[screenScales.length][0]; + this.screenScales = DoubleStream.of(screenScales).mapToObj(ScreenScale< T >::new).collect(Collectors.toList()); renderMaskArrays = new byte[0][]; - screenImages = new ArrayList<>(); - bufferedImages = new ArrayList<>(); - for (int i = 0; i < screenScales.length; ++i) - { - screenImages.add(Arrays.asList(null, null, null)); - bufferedImages.add(Arrays.asList(null, null, null)); - } - screenScaleTransforms = new AffineTransform3D[screenScales.length]; - pendingRepaintRequests = new Interval[screenScales.length]; - maxScreenScaleIndex = screenScales.length - 1; + maxScreenScaleIndex = this.screenScales.size() - 1; requestedScreenScaleIndex = maxScreenScaleIndex; } diff --git a/src/main/java/bdv/fx/viewer/render/RenderOutputImage.java b/src/main/java/bdv/fx/viewer/render/RenderOutputImage.java new file mode 100644 index 000000000..5b0f29837 --- /dev/null +++ b/src/main/java/bdv/fx/viewer/render/RenderOutputImage.java @@ -0,0 +1,23 @@ +package bdv.fx.viewer.render; + +import net.imglib2.img.array.ArrayImg; +import net.imglib2.img.basictypeaccess.IntAccess; +import net.imglib2.type.numeric.ARGBType; + +public interface RenderOutputImage { + + int width(); + + int height(); + + ArrayImg asArrayImg(); + + T unwrap(); + + interface Factory { + + RenderOutputImage< T > create( int width, int height ); + + RenderOutputImage< T > create( int width, int heihgt, RenderOutputImage< T > other); + } +}