192 lines
7.4 KiB
Java
192 lines
7.4 KiB
Java
package ctbrec.ui.controls;
|
|
|
|
import java.io.InterruptedIOException;
|
|
import java.util.Collections;
|
|
import java.util.List;
|
|
import java.util.Optional;
|
|
import java.util.concurrent.ExecutionException;
|
|
import java.util.concurrent.ExecutorService;
|
|
import java.util.concurrent.Executors;
|
|
import java.util.concurrent.Future;
|
|
|
|
import org.slf4j.Logger;
|
|
import org.slf4j.LoggerFactory;
|
|
|
|
import ctbrec.Config;
|
|
import ctbrec.Model;
|
|
import ctbrec.io.HttpException;
|
|
import ctbrec.recorder.download.StreamSource;
|
|
import javafx.application.Platform;
|
|
import javafx.geometry.Insets;
|
|
import javafx.scene.control.ProgressIndicator;
|
|
import javafx.scene.image.Image;
|
|
import javafx.scene.image.ImageView;
|
|
import javafx.scene.layout.Region;
|
|
import javafx.scene.layout.StackPane;
|
|
import javafx.scene.media.Media;
|
|
import javafx.scene.media.MediaPlayer;
|
|
import javafx.scene.media.MediaView;
|
|
|
|
public class StreamPreview extends StackPane {
|
|
private static final transient Logger LOG = LoggerFactory.getLogger(StreamPreview.class);
|
|
|
|
private ImageView preview = new ImageView();
|
|
private MediaView videoPreview;
|
|
private MediaPlayer videoPlayer;
|
|
private Media video;
|
|
private ProgressIndicator progressIndicator;
|
|
private static ExecutorService executor = Executors.newSingleThreadExecutor();
|
|
private static Future<?> future;
|
|
|
|
public StreamPreview() {
|
|
videoPreview = new MediaView();
|
|
videoPreview.setFitWidth(Config.getInstance().getSettings().thumbWidth);
|
|
videoPreview.setFitHeight(videoPreview.getFitWidth() * 9 / 16);
|
|
videoPreview.setPreserveRatio(true);
|
|
StackPane.setMargin(videoPreview, new Insets(5));
|
|
|
|
preview.setFitWidth(Config.getInstance().getSettings().thumbWidth);
|
|
preview.setPreserveRatio(true);
|
|
preview.setSmooth(true);
|
|
preview.setStyle("-fx-background-radius: 10px, 10px, 10px, 10px;");
|
|
preview.visibleProperty().bind(videoPreview.visibleProperty().not());
|
|
StackPane.setMargin(preview, new Insets(5));
|
|
|
|
progressIndicator = new ProgressIndicator();
|
|
progressIndicator.setVisible(false);
|
|
|
|
Region veil = new Region();
|
|
veil.setStyle("-fx-background-color: rgba(0, 0, 0, 0.8)");
|
|
veil.visibleProperty().bind(progressIndicator.visibleProperty());
|
|
StackPane.setMargin(veil, new Insets(5));
|
|
|
|
getChildren().addAll(preview, videoPreview, veil, progressIndicator);
|
|
}
|
|
|
|
public void startStream(Model model) {
|
|
Platform.runLater(() -> {
|
|
progressIndicator.setVisible(true);
|
|
if(model.getPreview() != null) {
|
|
try {
|
|
videoPreview.setVisible(false);
|
|
Image img = new Image(model.getPreview(), true);
|
|
preview.setImage(img);
|
|
double aspect = img.getWidth() / img.getHeight();
|
|
double w = Config.getInstance().getSettings().thumbWidth;
|
|
double h = w / aspect;
|
|
resizeTo(w, h);
|
|
} catch (Exception e) {}
|
|
}
|
|
});
|
|
if(future != null && !future.isDone()) {
|
|
future.cancel(true);
|
|
}
|
|
future = executor.submit(() -> {
|
|
try {
|
|
List<StreamSource> sources = model.getStreamSources();
|
|
Collections.sort(sources);
|
|
StreamSource best = sources.get(0);
|
|
checkInterrupt();
|
|
LOG.debug("Preview url for {} is {}", model.getName(), best.getMediaPlaylistUrl());
|
|
video = new Media(best.getMediaPlaylistUrl());
|
|
video.setOnError(() -> onError(videoPlayer));
|
|
if(videoPlayer != null) {
|
|
videoPlayer.dispose();
|
|
}
|
|
videoPlayer = new MediaPlayer(video);
|
|
videoPlayer.setMute(true);
|
|
checkInterrupt();
|
|
videoPlayer.setOnReady(() -> {
|
|
if(!future.isCancelled()) {
|
|
Platform.runLater(() -> {
|
|
double aspect = (double)video.getWidth() / video.getHeight();
|
|
double w = Config.getInstance().getSettings().thumbWidth;
|
|
double h = w / aspect;
|
|
resizeTo(w, h);
|
|
progressIndicator.setVisible(false);
|
|
videoPreview.setVisible(true);
|
|
videoPreview.setMediaPlayer(videoPlayer);
|
|
videoPlayer.play();
|
|
});
|
|
}
|
|
});
|
|
videoPlayer.setOnError(() -> onError(videoPlayer));
|
|
} catch (IllegalStateException e) {
|
|
if(e.getMessage().equals("Stream url unknown")) {
|
|
// fine hls url for mfc not known yet
|
|
} else {
|
|
LOG.warn("Couldn't start preview video: {}", e.getMessage());
|
|
}
|
|
showTestImage();
|
|
} catch (HttpException e) {
|
|
if(e.getResponseCode() != 404) {
|
|
LOG.warn("Couldn't start preview video: {}", e.getMessage());
|
|
}
|
|
showTestImage();
|
|
} catch (InterruptedException | InterruptedIOException e) {
|
|
// future has been canceled, that's fine
|
|
} catch (ExecutionException e) {
|
|
if(e.getCause() instanceof InterruptedException || e.getCause() instanceof InterruptedIOException) {
|
|
// future has been canceled, that's fine
|
|
} else {
|
|
LOG.warn("Couldn't start preview video: {}", e.getMessage());
|
|
}
|
|
showTestImage();
|
|
} catch (Exception e) {
|
|
LOG.warn("Couldn't start preview video: {}", e.getMessage());
|
|
showTestImage();
|
|
}
|
|
});
|
|
}
|
|
|
|
public void resizeTo(double w, double h) {
|
|
preview.setFitWidth(w);
|
|
preview.setFitHeight(h);
|
|
videoPreview.setFitWidth(w);
|
|
videoPreview.setFitHeight(h);
|
|
progressIndicator.setPrefSize(w, h);
|
|
}
|
|
|
|
public void stop() {
|
|
MediaPlayer old = videoPlayer;
|
|
Future<?> oldFuture = future;
|
|
new Thread(() -> {
|
|
if(oldFuture != null && !oldFuture.isDone()) {
|
|
oldFuture.cancel(true);
|
|
}
|
|
if(old != null) {
|
|
old.dispose();
|
|
}
|
|
}).start();
|
|
}
|
|
|
|
private void onError(MediaPlayer videoPlayer) {
|
|
LOG.error("Error while starting preview stream", videoPlayer.getError());
|
|
Optional<Throwable> cause = Optional.ofNullable(videoPlayer).map(v -> v.getError()).map(e -> e.getCause());
|
|
if(cause.isPresent()) {
|
|
LOG.error("Error while starting preview stream root cause:", cause.get());
|
|
}
|
|
showTestImage();
|
|
}
|
|
|
|
private void showTestImage() {
|
|
stop();
|
|
Platform.runLater(() -> {
|
|
videoPreview.setVisible(false);
|
|
Image img = new Image(getClass().getResource("/image_not_found.png").toString(), true);
|
|
preview.setImage(img);
|
|
double aspect = img.getWidth() / img.getHeight();
|
|
double w = Config.getInstance().getSettings().thumbWidth;
|
|
double h = w / aspect;
|
|
resizeTo(w, h);
|
|
progressIndicator.setVisible(false);
|
|
});
|
|
}
|
|
|
|
private void checkInterrupt() throws InterruptedException {
|
|
if(Thread.interrupted()) {
|
|
throw new InterruptedException();
|
|
}
|
|
}
|
|
}
|