mirror of
https://github.com/overcuriousity/autopsy-flatpak.git
synced 2025-07-19 19:14:55 +00:00
Merge pull request #1790 from millmanorama/consolidate_task_based_usage_of_ImageIO_low_level_API
Consolidate task based usage of image io low level api (after pr 1789)
This commit is contained in:
commit
4199d9287d
@ -35,7 +35,6 @@ import java.util.Collections;
|
|||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
import static java.util.Objects.isNull;
|
|
||||||
import static java.util.Objects.nonNull;
|
import static java.util.Objects.nonNull;
|
||||||
import java.util.SortedSet;
|
import java.util.SortedSet;
|
||||||
import java.util.TreeSet;
|
import java.util.TreeSet;
|
||||||
@ -347,22 +346,14 @@ public class ImageUtils {
|
|||||||
public static BufferedImage getThumbnail(Content content, int iconSize) {
|
public static BufferedImage getThumbnail(Content content, int iconSize) {
|
||||||
if (content instanceof AbstractFile) {
|
if (content instanceof AbstractFile) {
|
||||||
AbstractFile file = (AbstractFile) content;
|
AbstractFile file = (AbstractFile) content;
|
||||||
// If a thumbnail file is already saved locally
|
|
||||||
File cacheFile = getCachedThumbnailLocation(content.getId());
|
Task<javafx.scene.image.Image> thumbnailTask = newGetThumbnailTask(file, iconSize, true);
|
||||||
if (cacheFile.exists()) {
|
thumbnailTask.run();
|
||||||
try {
|
try {
|
||||||
BufferedImage thumbnail = ImageIO.read(cacheFile);
|
return SwingFXUtils.fromFXImage(thumbnailTask.get(), null);
|
||||||
if (isNull(thumbnail) || thumbnail.getWidth() != iconSize) {
|
} catch (InterruptedException | ExecutionException ex) {
|
||||||
return generateAndSaveThumbnail(file, iconSize, cacheFile);
|
LOGGER.log(Level.WARNING, "Failed to get thumbnail for {0}: " + ex.toString(), getContentPathSafe(content));
|
||||||
} else {
|
return DEFAULT_THUMBNAIL;
|
||||||
return thumbnail;
|
|
||||||
}
|
|
||||||
} catch (Exception ex) {
|
|
||||||
LOGGER.log(Level.WARNING, "ImageIO had a problem reading thumbnail for image {0}: {1}", new Object[]{content.getName(), ex.getLocalizedMessage()}); //NON-NLS //NOI18N
|
|
||||||
return generateAndSaveThumbnail(file, iconSize, cacheFile);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return generateAndSaveThumbnail(file, iconSize, cacheFile);
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
return DEFAULT_THUMBNAIL;
|
return DEFAULT_THUMBNAIL;
|
||||||
@ -513,87 +504,6 @@ public class ImageUtils {
|
|||||||
return fileHeaderBuffer;
|
return fileHeaderBuffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Generate an icon and save it to specified location.
|
|
||||||
*
|
|
||||||
* @param file File to generate icon for
|
|
||||||
* @param iconSize size in pixels of the thumbnail
|
|
||||||
* @param cacheFile Location to save thumbnail to
|
|
||||||
*
|
|
||||||
* @return Generated icon or null on error
|
|
||||||
*/
|
|
||||||
private static BufferedImage generateAndSaveThumbnail(AbstractFile file, int iconSize, File cacheFile) {
|
|
||||||
BufferedImage thumbnail = null;
|
|
||||||
try {
|
|
||||||
if (VideoUtils.isVideoThumbnailSupported(file)) {
|
|
||||||
if (openCVLoaded) {
|
|
||||||
thumbnail = VideoUtils.generateVideoThumbnail(file, iconSize);
|
|
||||||
} else {
|
|
||||||
return DEFAULT_THUMBNAIL;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
thumbnail = generateImageThumbnail(file, iconSize);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (thumbnail == null) {
|
|
||||||
return DEFAULT_THUMBNAIL;
|
|
||||||
|
|
||||||
} else {
|
|
||||||
BufferedImage toSave = thumbnail;
|
|
||||||
imageSaver.execute(() -> {
|
|
||||||
try {
|
|
||||||
Files.createParentDirs(cacheFile);
|
|
||||||
if (cacheFile.exists()) {
|
|
||||||
cacheFile.delete();
|
|
||||||
}
|
|
||||||
ImageIO.write(toSave, FORMAT, cacheFile);
|
|
||||||
} catch (IllegalArgumentException | IOException ex1) {
|
|
||||||
LOGGER.log(Level.WARNING, COULD_NOT_WRITE_CACHE_THUMBNAIL + file, ex1);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
} catch (NullPointerException ex) {
|
|
||||||
LOGGER.log(Level.WARNING, COULD_NOT_WRITE_CACHE_THUMBNAIL + file, ex);
|
|
||||||
}
|
|
||||||
return thumbnail;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Generate and return a scaled image
|
|
||||||
*
|
|
||||||
* @param content
|
|
||||||
* @param iconSize
|
|
||||||
*
|
|
||||||
* @return a Thumbnail of the given content at the given size, or null if
|
|
||||||
* there was a problem.
|
|
||||||
*/
|
|
||||||
@Nullable
|
|
||||||
private static BufferedImage generateImageThumbnail(AbstractFile content, int iconSize) {
|
|
||||||
|
|
||||||
try {
|
|
||||||
final ReadImageTask readImageTask = new ReadImageTask(content);
|
|
||||||
|
|
||||||
readImageTask.run();
|
|
||||||
BufferedImage bi = SwingFXUtils.fromFXImage(readImageTask.get(), null);
|
|
||||||
|
|
||||||
if (bi == null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
return ScalrWrapper.resizeFast(bi, iconSize);
|
|
||||||
} catch (IllegalArgumentException e) {
|
|
||||||
// if resizing does not work due to extreme aspect ratio,
|
|
||||||
// crop the image instead.
|
|
||||||
return ScalrWrapper.cropImage(bi, Math.min(iconSize, bi.getWidth()), Math.min(iconSize, bi.getHeight()));
|
|
||||||
}
|
|
||||||
} catch (OutOfMemoryError e) {
|
|
||||||
LOGGER.log(Level.WARNING, "Could not scale image (too large) " + content.getName() + ": " + e.toString()); //NON-NLS //NOI18N
|
|
||||||
} catch (Exception e) {
|
|
||||||
LOGGER.log(Level.WARNING, "ImageIO could not load image " + content.getName() + ": " + e.toString()); //NON-NLS //NOI18N
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the width of the given image, in pixels.
|
* Get the width of the given image, in pixels.
|
||||||
*
|
*
|
||||||
@ -711,8 +621,8 @@ public class ImageUtils {
|
|||||||
*
|
*
|
||||||
* @return a new Task that returns a thumbnail as its result.
|
* @return a new Task that returns a thumbnail as its result.
|
||||||
*/
|
*/
|
||||||
public static Task<javafx.scene.image.Image> newGetThumbnailTask(AbstractFile file, int iconSize) {
|
public static Task<javafx.scene.image.Image> newGetThumbnailTask(AbstractFile file, int iconSize, boolean defaultOnFailure) {
|
||||||
return new GetThumbnailTask(file, iconSize);
|
return new GetThumbnailTask(file, iconSize, defaultOnFailure);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -724,15 +634,17 @@ public class ImageUtils {
|
|||||||
|
|
||||||
private final int iconSize;
|
private final int iconSize;
|
||||||
private final File cacheFile;
|
private final File cacheFile;
|
||||||
|
private final boolean defaultOnFailure;
|
||||||
|
|
||||||
@NbBundle.Messages({"# {0} - file name",
|
@NbBundle.Messages({"# {0} - file name",
|
||||||
"GetOrGenerateThumbnailTask.loadingThumbnailFor=Loading thumbnail for {0}", "# {0} - file name",
|
"GetOrGenerateThumbnailTask.loadingThumbnailFor=Loading thumbnail for {0}", "# {0} - file name",
|
||||||
"GetOrGenerateThumbnailTask.generatingPreviewFor=Generating preview for {0}"})
|
"GetOrGenerateThumbnailTask.generatingPreviewFor=Generating preview for {0}"})
|
||||||
private GetThumbnailTask(AbstractFile file, int iconSize) {
|
private GetThumbnailTask(AbstractFile file, int iconSize, boolean defaultOnFailure) {
|
||||||
super(file);
|
super(file);
|
||||||
updateMessage(Bundle.GetOrGenerateThumbnailTask_loadingThumbnailFor(file.getName()));
|
updateMessage(Bundle.GetOrGenerateThumbnailTask_loadingThumbnailFor(file.getName()));
|
||||||
this.iconSize = iconSize;
|
this.iconSize = iconSize;
|
||||||
cacheFile = getCachedThumbnailLocation(file.getId());
|
this.defaultOnFailure = defaultOnFailure;
|
||||||
|
this.cacheFile = getCachedThumbnailLocation(file.getId());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -755,8 +667,10 @@ public class ImageUtils {
|
|||||||
if (openCVLoaded) {
|
if (openCVLoaded) {
|
||||||
updateMessage(Bundle.GetOrGenerateThumbnailTask_generatingPreviewFor(file.getName()));
|
updateMessage(Bundle.GetOrGenerateThumbnailTask_generatingPreviewFor(file.getName()));
|
||||||
thumbnail = VideoUtils.generateVideoThumbnail(file, iconSize);
|
thumbnail = VideoUtils.generateVideoThumbnail(file, iconSize);
|
||||||
} else {
|
} else if (defaultOnFailure) {
|
||||||
thumbnail = DEFAULT_THUMBNAIL;
|
thumbnail = DEFAULT_THUMBNAIL;
|
||||||
|
} else {
|
||||||
|
throw new IIOException("Failed to read image for thumbnail generation.");
|
||||||
}
|
}
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
|
@ -191,7 +191,7 @@ public enum ThumbnailCache {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
final Task<Image> newGetThumbnailTask = ImageUtils.newGetThumbnailTask(file.getAbstractFile(), MAX_THUMBNAIL_SIZE);
|
final Task<Image> newGetThumbnailTask = ImageUtils.newGetThumbnailTask(file.getAbstractFile(), MAX_THUMBNAIL_SIZE, false);
|
||||||
newGetThumbnailTask.stateProperty().addListener((Observable observable) -> {
|
newGetThumbnailTask.stateProperty().addListener((Observable observable) -> {
|
||||||
switch (newGetThumbnailTask.getState()) {
|
switch (newGetThumbnailTask.getState()) {
|
||||||
case SUCCEEDED:
|
case SUCCEEDED:
|
||||||
|
@ -39,6 +39,7 @@ import org.sleuthkit.autopsy.casemodule.Case;
|
|||||||
import org.sleuthkit.autopsy.coreutils.Logger;
|
import org.sleuthkit.autopsy.coreutils.Logger;
|
||||||
import org.sleuthkit.autopsy.imagegallery.FileTypeUtils;
|
import org.sleuthkit.autopsy.imagegallery.FileTypeUtils;
|
||||||
import org.sleuthkit.autopsy.imagegallery.ThumbnailCache;
|
import org.sleuthkit.autopsy.imagegallery.ThumbnailCache;
|
||||||
|
import org.sleuthkit.autopsy.imagegallery.utils.TaskUtils;
|
||||||
import org.sleuthkit.datamodel.AbstractFile;
|
import org.sleuthkit.datamodel.AbstractFile;
|
||||||
import org.sleuthkit.datamodel.BlackboardArtifact;
|
import org.sleuthkit.datamodel.BlackboardArtifact;
|
||||||
import org.sleuthkit.datamodel.BlackboardAttribute;
|
import org.sleuthkit.datamodel.BlackboardAttribute;
|
||||||
@ -281,7 +282,30 @@ public abstract class DrawableFile<T extends AbstractFile> extends AbstractFile
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public abstract Task<Image> getReadFullSizeImageTask();
|
public Task<Image> getReadFullSizeImageTask() {
|
||||||
|
Image image = (imageRef != null) ? imageRef.get() : null;
|
||||||
|
if (image == null || image.isError()) {
|
||||||
|
Task<Image> readImageTask = getReadFullSizeImageTaskHelper();
|
||||||
|
readImageTask.stateProperty().addListener(stateProperty -> {
|
||||||
|
switch (readImageTask.getState()) {
|
||||||
|
case SUCCEEDED:
|
||||||
|
try {
|
||||||
|
imageRef = new SoftReference<>(readImageTask.get());
|
||||||
|
} catch (InterruptedException | ExecutionException exception) {
|
||||||
|
LOGGER.log(Level.WARNING, getMessageTemplate(exception), getContentPathSafe());
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return readImageTask;
|
||||||
|
} else {
|
||||||
|
return TaskUtils.taskFrom(() -> image);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
abstract String getMessageTemplate(Exception exception);
|
||||||
|
|
||||||
|
abstract Task<Image> getReadFullSizeImageTaskHelper();
|
||||||
|
|
||||||
public void setAnalyzed(Boolean analyzed) {
|
public void setAnalyzed(Boolean analyzed) {
|
||||||
this.analyzed.set(analyzed);
|
this.analyzed.set(analyzed);
|
||||||
|
@ -19,9 +19,6 @@
|
|||||||
package org.sleuthkit.autopsy.imagegallery.datamodel;
|
package org.sleuthkit.autopsy.imagegallery.datamodel;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.lang.ref.SoftReference;
|
|
||||||
import java.util.concurrent.ExecutionException;
|
|
||||||
import javafx.beans.Observable;
|
|
||||||
import javafx.concurrent.Task;
|
import javafx.concurrent.Task;
|
||||||
import javafx.scene.image.Image;
|
import javafx.scene.image.Image;
|
||||||
import javax.imageio.ImageIO;
|
import javax.imageio.ImageIO;
|
||||||
@ -48,33 +45,13 @@ public class ImageFile<T extends AbstractFile> extends DrawableFile<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Task<Image> getReadFullSizeImageTask() {
|
String getMessageTemplate(final Exception exception) {
|
||||||
Image image = (imageRef != null) ? imageRef.get() : null;
|
return "Failed to read image {0}: " + exception.toString();
|
||||||
if (image == null || image.isError()) {
|
|
||||||
final Task<Image> newReadImageTask = ImageUtils.newReadImageTask(this.getAbstractFile());
|
|
||||||
newReadImageTask.stateProperty().addListener((Observable observable) -> {
|
|
||||||
switch (newReadImageTask.getState()) {
|
|
||||||
case CANCELLED:
|
|
||||||
break;
|
|
||||||
case FAILED:
|
|
||||||
break;
|
|
||||||
case SUCCEEDED:
|
|
||||||
try {
|
|
||||||
imageRef = new SoftReference<>(newReadImageTask.get());
|
|
||||||
} catch (InterruptedException | ExecutionException interruptedException) {
|
|
||||||
}
|
}
|
||||||
break;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return newReadImageTask;
|
|
||||||
} else {
|
|
||||||
return new Task<Image>() {
|
|
||||||
@Override
|
@Override
|
||||||
protected Image call() throws Exception {
|
Task<Image> getReadFullSizeImageTaskHelper() {
|
||||||
return image;
|
return ImageUtils.newReadImageTask(this.getAbstractFile());
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -19,27 +19,26 @@
|
|||||||
package org.sleuthkit.autopsy.imagegallery.datamodel;
|
package org.sleuthkit.autopsy.imagegallery.datamodel;
|
||||||
|
|
||||||
import com.google.common.io.Files;
|
import com.google.common.io.Files;
|
||||||
import java.awt.image.BufferedImage;
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.lang.ref.SoftReference;
|
import java.lang.ref.SoftReference;
|
||||||
import java.nio.file.Paths;
|
import java.nio.file.Paths;
|
||||||
import java.util.concurrent.ExecutionException;
|
|
||||||
import javafx.beans.Observable;
|
|
||||||
import javafx.concurrent.Task;
|
import javafx.concurrent.Task;
|
||||||
import javafx.embed.swing.SwingFXUtils;
|
|
||||||
import javafx.scene.image.Image;
|
import javafx.scene.image.Image;
|
||||||
import javafx.scene.media.Media;
|
import javafx.scene.media.Media;
|
||||||
import javafx.scene.media.MediaException;
|
import javafx.scene.media.MediaException;
|
||||||
import org.netbeans.api.progress.ProgressHandle;
|
import org.netbeans.api.progress.ProgressHandle;
|
||||||
import org.netbeans.api.progress.ProgressHandleFactory;
|
import org.netbeans.api.progress.ProgressHandleFactory;
|
||||||
import org.sleuthkit.autopsy.coreutils.ImageUtils;
|
import org.sleuthkit.autopsy.coreutils.ImageUtils;
|
||||||
|
import org.sleuthkit.autopsy.coreutils.Logger;
|
||||||
import org.sleuthkit.autopsy.coreutils.VideoUtils;
|
import org.sleuthkit.autopsy.coreutils.VideoUtils;
|
||||||
import org.sleuthkit.autopsy.datamodel.ContentUtils;
|
import org.sleuthkit.autopsy.datamodel.ContentUtils;
|
||||||
import org.sleuthkit.datamodel.AbstractFile;
|
import org.sleuthkit.datamodel.AbstractFile;
|
||||||
|
|
||||||
public class VideoFile<T extends AbstractFile> extends DrawableFile<T> {
|
public class VideoFile<T extends AbstractFile> extends DrawableFile<T> {
|
||||||
|
|
||||||
|
private static final Logger LOGGER = Logger.getLogger(VideoFile.class.getName());
|
||||||
|
|
||||||
private static final Image VIDEO_ICON = new Image("org/sleuthkit/autopsy/imagegallery/images/Clapperboard.png");
|
private static final Image VIDEO_ICON = new Image("org/sleuthkit/autopsy/imagegallery/images/Clapperboard.png");
|
||||||
|
|
||||||
VideoFile(T file, Boolean analyzed) {
|
VideoFile(T file, Boolean analyzed) {
|
||||||
@ -50,44 +49,16 @@ public class VideoFile<T extends AbstractFile> extends DrawableFile<T> {
|
|||||||
return VIDEO_ICON;
|
return VIDEO_ICON;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public Task<Image> getReadFullSizeImageTask() {
|
|
||||||
Image image = (imageRef != null) ? imageRef.get() : null;
|
|
||||||
if (image == null || image.isError()) {
|
|
||||||
Task<Image> newReadImageTask = new Task<Image>() {
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected Image call() throws Exception {
|
String getMessageTemplate(final Exception exception) {
|
||||||
final BufferedImage bufferedImage = ImageUtils.getThumbnail(getAbstractFile(), 1024);
|
return "Failed to get image preview for video {0}: " + exception.toString();
|
||||||
return (bufferedImage == ImageUtils.getDefaultThumbnail())
|
|
||||||
? null
|
|
||||||
: SwingFXUtils.toFXImage(bufferedImage, null);
|
|
||||||
}
|
}
|
||||||
};
|
|
||||||
|
|
||||||
newReadImageTask.stateProperty().addListener((Observable observable) -> {
|
|
||||||
switch (newReadImageTask.getState()) {
|
|
||||||
case CANCELLED:
|
|
||||||
break;
|
|
||||||
case FAILED:
|
|
||||||
break;
|
|
||||||
case SUCCEEDED:
|
|
||||||
try {
|
|
||||||
imageRef = new SoftReference<>(newReadImageTask.get());
|
|
||||||
} catch (InterruptedException | ExecutionException interruptedException) {
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return newReadImageTask;
|
|
||||||
} else {
|
|
||||||
return new Task<Image>() {
|
|
||||||
@Override
|
@Override
|
||||||
protected Image call() throws Exception {
|
Task<Image> getReadFullSizeImageTaskHelper() {
|
||||||
return image;
|
return ImageUtils.newGetThumbnailTask(getAbstractFile(), 1024, false);
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private SoftReference<Media> mediaRef;
|
private SoftReference<Media> mediaRef;
|
||||||
|
@ -0,0 +1,40 @@
|
|||||||
|
/*
|
||||||
|
* Autopsy Forensic Browser
|
||||||
|
*
|
||||||
|
* Copyright 2015 Basis Technology Corp.
|
||||||
|
* Contact: carrier <at> sleuthkit <dot> org
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.sleuthkit.autopsy.imagegallery.utils;
|
||||||
|
|
||||||
|
import java.util.concurrent.Callable;
|
||||||
|
import javafx.concurrent.Task;
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class TaskUtils {
|
||||||
|
|
||||||
|
public static <T> Task<T> taskFrom(Callable<T> callable) {
|
||||||
|
return new Task<T>() {
|
||||||
|
@Override
|
||||||
|
protected T call() throws Exception {
|
||||||
|
return callable.call();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private TaskUtils() {
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user