All Downloads are FREE. Search and download functionalities are using the official Maven repository.
Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
de.gsi.chart.utils.PeriodicScreenCapture Maven / Gradle / Ivy
package de.gsi.chart.utils;
import static java.nio.file.StandardCopyOption.REPLACE_EXISTING;
import java.io.File;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.LinkedList;
import java.util.List;
import java.util.Objects;
import java.util.TimeZone;
import java.util.Timer;
import java.util.TimerTask;
import java.util.concurrent.TimeUnit;
import javafx.animation.Animation;
import javafx.animation.KeyFrame;
import javafx.animation.Timeline;
import javafx.application.Platform;
import javafx.beans.InvalidationListener;
import javafx.beans.Observable;
import javafx.scene.Scene;
import javafx.scene.image.Image;
import javafx.scene.image.WritableImage;
import javafx.util.Duration;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Simple class to make a periodic (or on-demand) screen-shot of given JavaFX scene to file. Class permits to add an ISO
* date-time string
*
* @author rstein
*/
public class PeriodicScreenCapture implements Observable {
private static final Logger LOGGER = LoggerFactory.getLogger(PeriodicScreenCapture.class);
private static final String DEFAULT_TIME_FORMAT = "yyyyMMdd_HHmmss";
private static final String FILE_LOGGING_SUFFIX = ".png";
private final Scene primaryScene;
private final Path path;
private final String fileName;
private final double delay;
private final double period;
private Timeline periodicTask; // for JavaFX tasks
private String isoDateTimeFormatString = DEFAULT_TIME_FORMAT;
private final boolean addDateTime;
protected final List listeners = new LinkedList<>();
private final Timer timer = new Timer("sample-update-timer", true); // for non-JavaFX tasks
public PeriodicScreenCapture(final Path path, final String fileName, final Scene scene, final double delay,
final double period) {
this(path, fileName, scene, delay, period, false);
}
public PeriodicScreenCapture(final Path path, final String fileName, final Scene scene, final double delay,
final double period, final boolean addDateTime) {
this.path = path;
this.fileName = fileName.replace(".png", "").replace(".PNG", "");
primaryScene = scene;
this.delay = delay;
this.period = period;
this.addDateTime = addDateTime;
}
@Override
public void addListener(final InvalidationListener listener) {
Objects.requireNonNull(listener, "InvalidationListener must not be null");
// N.B. suppress duplicates
if (!listeners.contains(listener)) {
listeners.add(listener);
}
}
protected void executeFireInvalidated() {
for (final InvalidationListener listener : new ArrayList<>(listeners)) {
listener.invalidated(this);
}
}
public void fireInvalidated() {
if (listeners.isEmpty()) {
return;
}
if (Platform.isFxApplicationThread()) {
executeFireInvalidated();
} else {
Platform.runLater(this::executeFireInvalidated);
}
}
public String getIsoDateTimeFormatterString() {
return isoDateTimeFormatString;
}
public void performScreenCapture() {
try {
final WritableImage image = primaryScene.snapshot(null);
// open save in separate thread
timer.schedule(new TimerTask() {
@Override
public void run() {
writeImage(image);
}
}, 0);
LOGGER.debug("this is called periodic on UI thread");
} catch (final Exception e) {
// continue at all costs
LOGGER.error("error while writing screen captured image to file", e);
}
}
@Override
public void removeListener(final InvalidationListener listener) {
listeners.remove(listener);
}
public void setIsoDateTimeFormatterString(final String newFormat) {
if (newFormat == null || newFormat.isEmpty()) {
throw new IllegalArgumentException("new format must not be null or empty");
}
isoDateTimeFormatString = newFormat;
}
public void start() {
if (periodicTask != null) {
periodicTask.stop();
}
periodicTask = new Timeline(new KeyFrame(Duration.seconds(period), event -> performScreenCapture()));
periodicTask.setDelay(Duration.seconds(delay));
periodicTask.setCycleCount(Animation.INDEFINITE);
periodicTask.play();
}
public void stop() {
if (periodicTask != null) {
periodicTask.stop();
}
}
private void writeImage(final Image image) {
final long now = System.currentTimeMillis();
try {
final String format = getIsoDateTimeFormatterString();
final String longFileName = addDateTime && format != null && !format.isEmpty()
? path.toFile() + String.format("/%s_%s%s", fileName, getISODate(now, format), FILE_LOGGING_SUFFIX)
: path.toFile() + "/" + fileName;
final String tempFileName = longFileName + "_temp.png";
final File file = new File(tempFileName);
if (file.getParentFile() != null && file.getParentFile().mkdirs()) {
LOGGER.info("needed to create directory for file: " + longFileName);
}
WriteFxImage.savePng(image, file);
Files.move(Paths.get(tempFileName), Paths.get(longFileName), REPLACE_EXISTING);
fireInvalidated();
LOGGER.debug("write screenshot to " + tempFileName + " -> " + longFileName);
} catch (final Exception e) {
LOGGER.error("could not write to file: '" + fileName + "'", e);
}
}
protected static String getISODate(final long timeMillis, final String format) {
final long time = TimeUnit.MILLISECONDS.toMillis(timeMillis);
final TimeZone tz = TimeZone.getTimeZone("UTC");
final DateFormat df = new SimpleDateFormat(format);
df.setTimeZone(tz);
return df.format(new Date(time));
}
}