org.icepdf.ri.viewer.WindowManager Maven / Gradle / Ivy
/*
* Copyright 2006-2019 ICEsoft Technologies Canada Corp.
*
* 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.icepdf.ri.viewer;
import org.icepdf.core.pobjects.Document;
import org.icepdf.core.util.Defs;
import org.icepdf.ri.common.*;
import org.icepdf.ri.common.print.PrintHelper;
import org.icepdf.ri.common.views.Controller;
import org.icepdf.ri.common.views.DocumentViewController;
import org.icepdf.ri.common.views.DocumentViewControllerImpl;
import org.icepdf.ri.util.ViewerPropertiesManager;
import javax.swing.*;
import java.awt.*;
import java.net.URL;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.ResourceBundle;
import java.util.prefs.Preferences;
import static org.icepdf.ri.util.ViewerPropertiesManager.*;
/**
* An implementation of WindowManagementCallback to manage the viewer applications
* windows.
*
* @since 1.0
*/
public class WindowManager implements WindowManagementCallback {
public static final String APPLICATION_WIDTH = "application.width";
public static final String APPLICATION_HEIGHT = "application.height";
public static final String APPLICATION_X_OFFSET = "application.x";
public static final String APPLICATION_Y_OFFSET = "application.y";
public static final int NEW_WINDOW_OFFSET = 10;
private static WindowManager windowManager;
private ViewerPropertiesManager properties;
private ArrayList controllers;
private static int newWindowInvocationCounter = 0;
private ResourceBundle messageBundle = null;
public static ControllerFactory controllerFactory = SwingControllerFactory.getInstance();
public static final ViewBuilderFactory viewBuilderFactory = SwingViewBuilderFactory.getInstance();
private WindowManager() {
}
public static WindowManager getInstance() {
return windowManager;
}
//window management functions
public static WindowManager createInstance(ViewerPropertiesManager properties, ResourceBundle messageBundle) {
windowManager = new WindowManager();
windowManager.properties = properties;
windowManager.controllers = new ArrayList<>();
if (messageBundle != null) {
windowManager.messageBundle = messageBundle;
} else {
windowManager.messageBundle = ResourceBundle.getBundle(
ViewerPropertiesManager.DEFAULT_MESSAGE_BUNDLE);
}
// Announce ourselves...
if (Defs.booleanProperty("org.icepdf.core.verbose", true)) {
System.out.println("\nICEpdf Viewer " + Document.getLibraryVersion());
}
return windowManager;
}
public ViewerPropertiesManager getProperties() {
return properties;
}
public long getNumberOfWindows() {
return newWindowInvocationCounter;
}
public Collection getControllers() {
return controllers;
}
public void newWindow(final String location) {
Controller controller = commonWindowCreation();
controller.openDocument(location);
}
public void newWindow(final Document document, final String fileName) {
Controller controller = commonWindowCreation();
controller.openDocument(document, fileName);
}
public void newWindow(URL location) {
Controller controller = commonWindowCreation();
controller.openDocument(location);
}
public void newWindow(final String location, final String printer) {
Controller controller = commonWindowCreation(false);
controller.openDocument(location);
print(controller, printer);
}
public void newWindow(final Document document, final String fileName, final String printer) {
Controller controller = commonWindowCreation(false);
controller.openDocument(document, fileName);
print(controller, printer);
}
public void newWindow(URL location, final String printer) {
Controller controller = commonWindowCreation(false);
controller.openDocument(location);
print(controller, printer);
}
private void print(Controller controller, String printer) {
controller.printAndExit(!PrintHelper.hasPrinter(printer), printer);
quit(controller, controller.getViewerFrame(), controller.getPropertiesManager().getPreferences());
}
protected Controller commonWindowCreation() {
return commonWindowCreation(true);
}
protected Controller commonWindowCreation(boolean isVisible) {
Controller controller = new SwingController(messageBundle);
controller.setWindowManagementCallback(this);
// add interactive mouse link annotation support
controller.getDocumentViewController().setAnnotationCallback(
new MyAnnotationCallback(controller.getDocumentViewController()));
controllers.add(controller);
// guild a new swing viewer with remembered view settings.
int viewType = DocumentViewControllerImpl.ONE_PAGE_VIEW;
int pageFit = DocumentViewController.PAGE_FIT_WINDOW_WIDTH;
float pageRotation = 0;
Preferences viewerPreferences = getProperties().getPreferences();
try {
viewType = viewerPreferences.getInt(PROPERTY_DEFAULT_VIEW_TYPE,
DocumentViewControllerImpl.ONE_PAGE_VIEW);
pageFit = viewerPreferences.getInt(ViewerPropertiesManager.PROPERTY_DEFAULT_PAGEFIT,
DocumentViewController.PAGE_FIT_WINDOW_WIDTH);
pageRotation = viewerPreferences.getFloat(ViewerPropertiesManager.PROPERTY_DEFAULT_ROTATION, pageRotation);
} catch (NumberFormatException e) {
// eating error, as we can continue with out alarm
}
ViewBuilder factory = viewBuilderFactory.create(controller, viewType, pageFit, pageRotation);
JFrame frame = factory.buildViewerFrame();
if (frame != null) {
newWindowLocation(frame);
frame.setVisible(isVisible);
}
return controller;
}
/**
* Loads the last used windows location as well as other frame related settings and insures the frame is
* visible.
*
* @param frame parent window containers.
*/
public static void newWindowLocation(Container frame) {
GraphicsEnvironment env = GraphicsEnvironment.getLocalGraphicsEnvironment();
Rectangle bounds = env.getMaximumWindowBounds();
Preferences prefs = ViewerPropertiesManager.getInstance().getPreferences();
// get the last used window size.
int width = prefs.getInt(APPLICATION_WIDTH, 800);
int height = prefs.getInt(APPLICATION_HEIGHT, 600);
// default center for width on primary screen.
int x = (int) (bounds.getMinX() + bounds.getWidth() / 2 - width / 2);
int y = (int) (bounds.getMinY() + bounds.getHeight() / 2 - height / 2);
// get teh last used window offset.
int previousX = prefs.getInt(APPLICATION_X_OFFSET, x);
int previousY = prefs.getInt(APPLICATION_Y_OFFSET, y);
// quick check to make sure the viewer will be visible in at least one screen, if not we default to primary
GraphicsDevice[] graphicsDevices = env.getScreenDevices();
ArrayList results = new ArrayList<>();
for (GraphicsDevice screen : graphicsDevices) {
GraphicsConfiguration config = screen.getDefaultConfiguration();
if (config.getBounds().contains(previousX, previousY, width, height)) {
results.add(screen);
}
}
// no intersection with a current screen then we go with the centering coordinates of the primary screen.
if (results.size() == 0) {
previousX = x;
previousY = y;
}
prefs.putInt(APPLICATION_X_OFFSET, previousX);
prefs.putInt(APPLICATION_Y_OFFSET, previousY);
// apply the corrected size and location.
frame.setSize(width, height);
int offset = newWindowInvocationCounter > 0 ? NEW_WINDOW_OFFSET : 0;
frame.setLocation(previousX + offset, previousY + offset);
newWindowInvocationCounter++;
}
public static void saveViewerState(Container viewer) {
if (viewer != null) {
//save width & height
Rectangle sz = viewer.getBounds();
Preferences viewerPreferences = ViewerPropertiesManager.getInstance().getPreferences();
viewerPreferences.putInt(APPLICATION_X_OFFSET, sz.x);
viewerPreferences.putInt(APPLICATION_Y_OFFSET, sz.y);
viewerPreferences.putInt(APPLICATION_WIDTH, sz.width);
viewerPreferences.putInt(APPLICATION_HEIGHT, sz.height);
viewerPreferences.putInt(ViewerPropertiesManager.PROPERTY_DEFAULT_PAGEFIT,
viewerPreferences.getInt(ViewerPropertiesManager.PROPERTY_DEFAULT_PAGEFIT, 0));
int viewType = viewerPreferences.getInt(PROPERTY_DEFAULT_VIEW_TYPE, 1);
// don't save the attachments view as it only applies to specific
// document types.
if (viewType != DocumentViewControllerImpl.USE_ATTACHMENTS_VIEW) {
viewerPreferences.putInt(PROPERTY_DEFAULT_VIEW_TYPE,
viewerPreferences.getInt(PROPERTY_DEFAULT_VIEW_TYPE, 1));
}
if (ViewModel.getDefaultFilePath() != null) {
viewerPreferences.put(PROPERTY_DEFAULT_FILE_PATH, ViewModel.getDefaultFilePath());
}
if (ViewModel.getDefaultURL() != null) {
viewerPreferences.put(PROPERTY_DEFAULT_URL, ViewModel.getDefaultURL());
}
}
}
public void disposeWindow(Controller controller, Frame viewer,
Preferences preferences) {
if (controllers.size() <= 1) {
quit(controller, viewer, preferences);
return;
}
//gets the window to close from the list
int index = controllers.indexOf(controller);
if (index >= 0) {
controllers.remove(index);
newWindowInvocationCounter--;
if (viewer != null) {
viewer.setVisible(false);
viewer.dispose();
}
}
}
public void quit(Controller controller, Frame viewer,
Preferences preferences) {
saveViewerState(viewer);
// make sure all the controllers have been disposed.
for (Controller c : controllers) {
if (c == null)
continue;
c.dispose();
}
System.exit(0);
}
public void minimiseAllWindows() {
for (Controller controller : controllers) {
Frame frame = controller.getViewerFrame();
if (frame != null)
frame.setState(Frame.ICONIFIED);
}
}
public void bringAllWindowsToFront(Controller frontMost) {
Frame frontMostFrame = null;
for (Controller controller : controllers) {
Frame frame = controller.getViewerFrame();
if (frame != null) {
if (frontMost == controller) {
frontMostFrame = frame;
continue;
}
frame.setState(Frame.NORMAL);
frame.toFront();
}
}
if (frontMostFrame != null) {
frontMostFrame.setState(Frame.NORMAL);
frontMostFrame.toFront();
}
}
public void bringWindowToFront(int index) {
if (index >= 0 && index < controllers.size()) {
Controller controller = controllers.get(index);
Frame frame = controller.getViewerFrame();
if (frame != null) {
frame.setState(Frame.NORMAL);
frame.toFront();
}
}
}
/**
* As long as no windows have opened or closed, then the indexes in the
* returned list should still be valid for doing operations on
* the respective Controller objects
*
* @param giveIndex Give this SwingControllers index in the list as an Integer appended to the List
* @return List of String objects, each representing an open Document's origin. The last element may be an Integer
*/
public List getWindowDocumentOriginList(Controller giveIndex) {
Integer foundIndex = null;
int count = controllers.size();
List
© 2015 - 2025 Weber Informatics LLC | Privacy Policy