|
@@ -6,7 +6,6 @@ import org.slf4j.Logger;
|
|
|
import org.slf4j.LoggerFactory;
|
|
|
|
|
|
import javax.inject.Inject;
|
|
|
-import javafx.beans.binding.Bindings;
|
|
|
import javafx.beans.binding.BooleanBinding;
|
|
|
import javafx.collections.ObservableList;
|
|
|
import javafx.fxml.FXML;
|
|
@@ -15,6 +14,7 @@ import javafx.scene.input.MouseEvent;
|
|
|
import javafx.scene.layout.Region;
|
|
|
import javafx.stage.Screen;
|
|
|
import javafx.stage.Stage;
|
|
|
+import javafx.stage.WindowEvent;
|
|
|
|
|
|
@MainWindow
|
|
|
public class ResizeController implements FxController {
|
|
@@ -53,48 +53,70 @@ public class ResizeController implements FxController {
|
|
|
public void initialize() {
|
|
|
LOG.trace("init ResizeController");
|
|
|
|
|
|
- if (neverTouched()) {
|
|
|
- settings.displayConfiguration.set(getMonitorSizes());
|
|
|
- return;
|
|
|
- } else {
|
|
|
- if (didDisplayConfigurationChange()) {
|
|
|
- //If the position is illegal, then the window appears on the main screen in the middle of the window.
|
|
|
- Rectangle2D primaryScreenBounds = Screen.getPrimary().getBounds();
|
|
|
- window.setX((primaryScreenBounds.getWidth() - window.getMinWidth()) / 2);
|
|
|
- window.setY((primaryScreenBounds.getHeight() - window.getMinHeight()) / 2);
|
|
|
- window.setWidth(window.getMinWidth());
|
|
|
- window.setHeight(window.getMinHeight());
|
|
|
- } else {
|
|
|
- window.setHeight(settings.windowHeight.get() > window.getMinHeight() ? settings.windowHeight.get() : window.getMinHeight());
|
|
|
- window.setWidth(settings.windowWidth.get() > window.getMinWidth() ? settings.windowWidth.get() : window.getMinWidth());
|
|
|
- window.setX(settings.windowXPosition.get());
|
|
|
- window.setY(settings.windowYPosition.get());
|
|
|
- }
|
|
|
+ if (!neverTouched()) {
|
|
|
+ window.setHeight(settings.windowHeight.get() > window.getMinHeight() ? settings.windowHeight.get() : window.getMinHeight());
|
|
|
+ window.setWidth(settings.windowWidth.get() > window.getMinWidth() ? settings.windowWidth.get() : window.getMinWidth());
|
|
|
+ window.setX(settings.windowXPosition.get());
|
|
|
+ window.setY(settings.windowYPosition.get());
|
|
|
}
|
|
|
- savePositionalSettings();
|
|
|
+
|
|
|
+ window.setOnShowing(this::checkDisplayBounds);
|
|
|
}
|
|
|
|
|
|
private boolean neverTouched() {
|
|
|
return (settings.windowHeight.get() == 0) && (settings.windowWidth.get() == 0) && (settings.windowXPosition.get() == 0) && (settings.windowYPosition.get() == 0);
|
|
|
}
|
|
|
|
|
|
- private boolean didDisplayConfigurationChange() {
|
|
|
- String currentDisplayConfiguration = getMonitorSizes();
|
|
|
- String settingsDisplayConfiguration = settings.displayConfiguration.get();
|
|
|
- boolean configurationHasChanged = !settingsDisplayConfiguration.equals(currentDisplayConfiguration);
|
|
|
- if (configurationHasChanged) settings.displayConfiguration.set(currentDisplayConfiguration);
|
|
|
- return configurationHasChanged;
|
|
|
+ private boolean isWithinDisplayBounds() {
|
|
|
+ // (x1, y1) is the top left corner of the window, (x2, y2) is the bottom right corner
|
|
|
+ final double slack = 10;
|
|
|
+ final double width = window.getWidth() - 2 * slack;
|
|
|
+ final double height = window.getHeight() - 2 * slack;
|
|
|
+ final double x1 = window.getX() + slack;
|
|
|
+ final double y1 = window.getY() + slack;
|
|
|
+ final double x2 = x1 + width;
|
|
|
+ final double y2 = y1 + height;
|
|
|
+
|
|
|
+ final ObservableList<Screen> screens = Screen.getScreensForRectangle(x1, y1, width, height);
|
|
|
+
|
|
|
+ // Find the total visible area of the window
|
|
|
+ double visibleArea = 0;
|
|
|
+ for (Screen screen : screens) {
|
|
|
+ Rectangle2D bounds = screen.getVisualBounds();
|
|
|
+
|
|
|
+ double xOverlap = Math.min(x2, bounds.getMaxX()) - Math.max(x1, bounds.getMinX());
|
|
|
+ double yOverlap = Math.min(y2, bounds.getMaxY()) - Math.max(y1, bounds.getMinY());
|
|
|
+
|
|
|
+ visibleArea += xOverlap * yOverlap;
|
|
|
+ }
|
|
|
+
|
|
|
+ final double windowArea = width * height;
|
|
|
+
|
|
|
+ // Within bounds if the visible area matches the window area
|
|
|
+ return visibleArea == windowArea;
|
|
|
}
|
|
|
|
|
|
- private String getMonitorSizes() {
|
|
|
- ObservableList<Screen> screens = Screen.getScreens();
|
|
|
- StringBuilder sb = new StringBuilder();
|
|
|
- for (int i = 0; i < screens.size(); i++) {
|
|
|
- Rectangle2D screenBounds = screens.get(i).getBounds();
|
|
|
- if (!sb.isEmpty()) sb.append(" ");
|
|
|
- sb.append("displayId: " + i + ", " + screenBounds.getWidth() + "x" + screenBounds.getHeight() + ";");
|
|
|
+ private void checkDisplayBounds(WindowEvent evt) {
|
|
|
+
|
|
|
+ // Minimizing a window in Windows and closing it could result in an out of bounds position at (x, y) = (-32000, -32000)
|
|
|
+ // See https://devblogs.microsoft.com/oldnewthing/20041028-00/?p=37453
|
|
|
+ // If the position is (-32000, -32000), restore to the last saved position
|
|
|
+ if (window.getX() == -32000 && window.getY() == -32000) {
|
|
|
+ window.setX(settings.windowXPosition.get());
|
|
|
+ window.setY(settings.windowYPosition.get());
|
|
|
+ window.setWidth(settings.windowWidth.get());
|
|
|
+ window.setHeight(settings.windowHeight.get());
|
|
|
+ }
|
|
|
+
|
|
|
+ if (!isWithinDisplayBounds()) {
|
|
|
+ // If the position is illegal, then the window appears on the main screen in the middle of the window.
|
|
|
+ Rectangle2D primaryScreenBounds = Screen.getPrimary().getBounds();
|
|
|
+ window.setX((primaryScreenBounds.getWidth() - window.getMinWidth()) / 2);
|
|
|
+ window.setY((primaryScreenBounds.getHeight() - window.getMinHeight()) / 2);
|
|
|
+ window.setWidth(window.getMinWidth());
|
|
|
+ window.setHeight(window.getMinHeight());
|
|
|
+ savePositionalSettings();
|
|
|
}
|
|
|
- return sb.toString();
|
|
|
}
|
|
|
|
|
|
private void startResize(MouseEvent evt) {
|
|
@@ -183,5 +205,4 @@ public class ResizeController implements FxController {
|
|
|
public boolean isShowResizingArrows() {
|
|
|
return showResizingArrows.get();
|
|
|
}
|
|
|
-
|
|
|
}
|