All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.icepdf.ri.util.ViewerPropertiesManager 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.util;

import javax.swing.*;
import java.io.*;
import java.net.URL;
import java.util.*;
import java.util.logging.Level;
import java.util.logging.Logger;
import java.util.prefs.BackingStoreException;
import java.util.prefs.InvalidPreferencesFormatException;
import java.util.prefs.Preferences;


/**
 * 

This class provides a wrapper for persisting the viewer ri's settings. The class also provides mechanisms for * loading default properties. The class is divided into two sub concepts; the first being local properties (not sticky) * that are only persisted by command and the second being a preferences object that automatically saves values (sticky). *

*

Local Properties * The SwingViewBuilder uses local properties to build the AWT Viewer. The viewer will always have a default state * which can be altered with the checkAndStore*Property() methods. A given viewer configuration state can be persisted * to the preference object with a call to saveStoreProperties(). Once local properties are persisted the only way * to restore the default state is to call clearPreferences() or configure a customer ICEpdfDefault.properties file. *

* *

Preferences * The preferences object can be directly accessed via the accessor to directly access the preference api. *

* *

Default Properties * The default viewer ri ships with a ICEpdfDefault.properties file that can be used to load default viewer properties * for custom builds. This mechanism can be easier to implement then manually manipulating the PropertyManagers before * the a swing view is created. *

* * @since 6.3 */ public final class ViewerPropertiesManager { private static final Logger logger = Logger.getLogger(ViewerPropertiesManager.class.toString()); // use ascii '27' or ESC as the delimiting character when storing multiple values in one property name. public static final String PROPERTY_TOKEN_SEPARATOR = "|"; //default file for all not specified properties public static final String DEFAULT_PROP_FILE = "ICEpdfDefault.properties"; public static final String DEFAULT_PROP_FILE_PATH = "org/icepdf/ri/viewer/res/"; public static final String DEFAULT_MESSAGE_BUNDLE = "org.icepdf.ri.resources.MessageBundle"; public static final String PROPERTY_DEFAULT_FILE_PATH = "application.default.filepath"; public static final String PROPERTY_DEFAULT_URL = "application.default.url"; public static final String PROPERTY_RECENT_FILES_SIZE = "application.menu.recent.file.size"; public static final String PROPERTY_ICON_DEFAULT_SIZE = "application.icon.default.size"; // window properties public static final String PROPERTY_DIVIDER_LOCATION = "application.divider.location"; // default page fit mode public static final String PROPERTY_DEFAULT_PAGEFIT = "document.pagefit.mode"; public static final String PROPERTY_DEFAULT_ROTATION = "document.rotation"; // page rotation public static final String PROPERTY_DEFAULT_VIEW_TYPE = "document.viewtype"; // display tool public static final String PROPERTY_DEFAULT_DISPLAY_TOOL = "document.displaytool.mode"; // highlight and selection text colours. public static final String PROPERTY_TEXT_SELECTION_COLOR = "org.icepdf.core.views.page.text.selection.color"; public static final String PROPERTY_TEXT_HIGHLIGHT_COLOR = "org.icepdf.core.views.page.text.highlight.color"; // page view colour settings. public static final String PROPERTY_PAGE_VIEW_SHADOW_COLOR = "org.icepdf.core.views.page.shadow.color"; public static final String PROPERTY_PAGE_VIEW_PAPER_COLOR = "org.icepdf.core.views.page.paper.color"; public static final String PROPERTY_PAGE_VIEW_BORDER_COLOR = "org.icepdf.core.views.page.border.color"; public static final String PROPERTY_PAGE_VIEW_BACKGROUND_COLOR = "org.icepdf.core.views.background.color"; // image reference type. public static final String PROPERTY_IMAGING_REFERENCE_TYPE = "org.icepdf.core.imageReference"; // advanced threading properties public static final String PROPERTY_IMAGE_PROXY_ENABLED = "org.icepdf.core.imageProxy"; public static final String PROPERTY_IMAGE_PROXY_THREAD_COUNT = "org.icepdf.core.library.imageThreadPoolSize"; public static final String PROPERTY_COMMON_THREAD_COUNT = "org.icepdf.core.library.threadPoolSize"; // properties used to hide/show toolbars public static final String PROPERTY_SHOW_MENU_RECENT_FILES = "application.toolbar.show.resentfiles"; public static final String PROPERTY_SHOW_TOOLBAR_UTILITY = "application.toolbar.show.utility"; public static final String PROPERTY_SHOW_TOOLBAR_PAGENAV = "application.toolbar.show.pagenav"; public static final String PROPERTY_SHOW_TOOLBAR_ZOOM = "application.toolbar.show.zoom"; public static final String PROPERTY_SHOW_TOOLBAR_FIT = "application.toolbar.show.fit"; public static final String PROPERTY_SHOW_TOOLBAR_FULL_SCREEN = "application.toolbar.show.fullscreen"; public static final String PROPERTY_SHOW_TOOLBAR_ROTATE = "application.toolbar.show.rotate"; public static final String PROPERTY_SHOW_TOOLBAR_TOOL = "application.toolbar.show.tool"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION = "application.toolbar.show.annotation"; public static final String PROPERTY_SHOW_TOOLBAR_FORMS = "application.toolbar.show.forms"; public static final String PROPERTY_SHOW_TOOLBAR_SEARCH = "application.toolbar.show.search"; // properties used to hide/show status bar buttons public static final String PROPERTY_SHOW_STATUSBAR = "application.statusbar"; // properties used to hide/show status bar status label public static final String PROPERTY_SHOW_STATUSBAR_STATUSLABEL = "application.statusbar.show.statuslabel"; // properties used to hide/show status bar buttons public static final String PROPERTY_SHOW_STATUSBAR_VIEWMODE = "application.statusbar.show.viewmode"; public static final String PROPERTY_SHOW_STATUSBAR_VIEWMODE_SINGLE = "application.statusbar.show.viewmode.singlepage"; public static final String PROPERTY_SHOW_STATUSBAR_VIEWMODE_SINGLE_CONTINUOUS = "application.statusbar.show.viewmode.single.page.continuous"; public static final String PROPERTY_SHOW_STATUSBAR_VIEWMODE_DOUBLE = "application.statusbar.show.viewmode.double.page"; public static final String PROPERTY_SHOW_STATUSBAR_VIEWMODE_DOUBLE_CONTINUOUS = "application.statusbar.show.viewmode.double.page.continuous"; // properties used to hide/show the utility buttons (open, print, etc.) public static final String PROPERTY_SHOW_UTILITY_OPEN = "application.toolbar.show.utility.open"; public static final String PROPERTY_SHOW_UTILITY_SAVE = "application.toolbar.show.utility.save"; public static final String PROPERTY_SHOW_UTILITY_PRINT = "application.toolbar.show.utility.print"; public static final String PROPERTY_SHOW_UTILITY_SEARCH = "application.toolbar.show.utility.search"; public static final String PROPERTY_SHOW_UTILITY_UPANE = "application.toolbar.show.utility.upane"; // properties used to hide/show utility pane tabs public static final String PROPERTY_HIDE_UTILITYPANE = "application.utilitypane.hide"; public static final String PROPERTY_SHOW_UTILITYPANE_BOOKMARKS = "application.utilitypane.show.bookmarks"; public static final String PROPERTY_SHOW_UTILITYPANE_ATTACHMENTS = "application.utilitypane.show.attachments"; public static final String PROPERTY_SHOW_UTILITYPANE_SEARCH = "application.utilitypane.show.search"; public static final String PROPERTY_SHOW_UTILITYPANE_THUMBNAILS = "application.utilitypane.show.thumbs"; public static final String PROPERTY_SHOW_UTILITYPANE_LAYERS = "application.utilitypane.show.layers"; public static final String PROPERTY_SHOW_UTILITYPANE_ANNOTATION = "application.utilitypane.show.annotation"; public static final String PROPERTY_SHOW_UTILITYPANE_ANNOTATION_FLAGS = "application.utilitypane.show.annotation.flags"; public static final String PROPERTY_SHOW_UTILITYPANE_SIGNATURES = "application.utilitypane.show.signatures"; // sub control for annotation tabs public static final String PROPERTY_SHOW_UTILITYPANE_ANNOTATION_MARKUP = "application.utilitypane.show.annotation.markup"; public static final String PROPERTY_SHOW_UTILITYPANE_ANNOTATION_DESTINATIONS = "application.utilitypane.show.annotation.dests"; // properties use dot hide/show preferences pane tabs. public static final String PROPERTY_SHOW_PREFERENCES_GENERAL = "application.preferences.show.general"; public static final String PROPERTY_SHOW_PREFERENCES_ANNOTATIONS = "application.preferences.show.annotations"; public static final String PROPERTY_SHOW_PREFERENCES_IMAGING = "application.preferences.show.imaging"; public static final String PROPERTY_SHOW_PREFERENCES_FONTS = "application.preferences.show.fonts"; public static final String PROPERTY_SHOW_PREFERENCES_ADVANCED = "application.preferences.show.advanced"; public static final String PROPERTY_SHOW_PREFERENCES_EXIMPORT = "application.preferences.show.eximport"; // default utility pane thumbnail zoom size for non-embedded files public static final String PROPERTY_UTILITYPANE_THUMBNAILS_ZOOM = "application.utilitypane.thumbnail.zoom"; // properties used for default zoom levels public static final String PROPERTY_DEFAULT_ZOOM_LEVEL = "application.zoom.factor.default"; public static final String PROPERTY_ZOOM_RANGES = "application.zoom.range.default"; // property to hide/show menu keyboard accelerator shortcuts public static final String PROPERTY_SHOW_KEYBOARD_SHORTCUTS = "application.menuitem.show.keyboard.shortcuts"; // properties used for overriding ViewerPreferences pulled from the document public static final String PROPERTY_VIEWPREF_HIDETOOLBAR = "application.viewerpreferences.hidetoolbar"; public static final String PROPERTY_VIEWPREF_HIDEMENUBAR = "application.viewerpreferences.hidemenubar"; public static final String PROPERTY_VIEWPREF_FITWINDOW = "application.viewerpreferences.fitwindow"; public static final String PROPERTY_VIEWPREF_FORM_HIGHLIGHT = "application.viewerpreferences.form.highlight"; public static final String PROPERTY_VIEWPREF_ANNOTATION_EDIT_MODE = "application.viewerpreferences.annotation.editmode"; // annotation handler default to selected tool after annotation is created. public static final String PROPERTY_ANNOTATION_HIGHLIGHT_SELECTION_TYPE = "application.annotation.highlight.selection.type"; public static final String PROPERTY_ANNOTATION_REDACTION_SELECTION_TYPE = "application.annotation.redaction" + ".selection.type"; public static final String PROPERTY_ANNOTATION_LINE_SELECTION_TYPE = "application.annotation.line.selection.type"; public static final String PROPERTY_ANNOTATION_LINK_SELECTION_TYPE = "application.annotation.link.selection.type"; public static final String PROPERTY_ANNOTATION_SQUARE_SELECTION_TYPE = "application.annotation.rectangle.selection.type"; public static final String PROPERTY_ANNOTATION_CIRCLE_SELECTION_TYPE = "application.annotation.circle.selection.type"; public static final String PROPERTY_ANNOTATION_INK_SELECTION_TYPE = "application.annotation.ink.selection.type"; public static final String PROPERTY_ANNOTATION_FREE_TEXT_SELECTION_TYPE = "application.annotation.freetext.selection.type"; public static final String PROPERTY_ANNOTATION_TEXT_SELECTION_TYPE = "application.annotation.text.selection.type"; public static final Set ALL_SELECTION_PROPERTIES = new HashSet<>(Arrays.asList( ViewerPropertiesManager.PROPERTY_ANNOTATION_CIRCLE_SELECTION_TYPE, ViewerPropertiesManager.PROPERTY_ANNOTATION_HIGHLIGHT_SELECTION_TYPE, ViewerPropertiesManager.PROPERTY_ANNOTATION_INK_SELECTION_TYPE, ViewerPropertiesManager.PROPERTY_ANNOTATION_LINE_SELECTION_TYPE, ViewerPropertiesManager.PROPERTY_ANNOTATION_LINK_SELECTION_TYPE, ViewerPropertiesManager.PROPERTY_ANNOTATION_SQUARE_SELECTION_TYPE, ViewerPropertiesManager.PROPERTY_ANNOTATION_TEXT_SELECTION_TYPE, ViewerPropertiesManager.PROPERTY_ANNOTATION_FREE_TEXT_SELECTION_TYPE )); // properties used to control visibility of annotation controls on main utility panel. public static final String PROPERTY_ANNOTATION_PROPERTIES_HIGHLIGHT_ENABLED = "application.annotation.properties.highlight.enabled"; public static final String PROPERTY_ANNOTATION_PROPERTIES_UNDERLINE_ENABLED = "application.annotation.properties.underline.enabled"; public static final String PROPERTY_ANNOTATION_PROPERTIES_STRIKE_OUT_ENABLED = "application.annotation.properties.strikeout.enabled"; public static final String PROPERTY_ANNOTATION_PROPERTIES_LINE_ENABLED = "application.annotation.properties.line.enabled"; public static final String PROPERTY_ANNOTATION_PROPERTIES_LINK_ENABLED = "application.annotation.properties.link.enabled"; public static final String PROPERTY_ANNOTATION_PROPERTIES_ARROW_ENABLED = "application.annotation.properties.arrow.enabled"; public static final String PROPERTY_ANNOTATION_PROPERTIES_RECTANGLE_ENABLED = "application.annotation.properties.rectangle.enabled"; public static final String PROPERTY_ANNOTATION_PROPERTIES_CIRCLE_ENABLED = "application.annotation.properties.circle.enabled"; public static final String PROPERTY_ANNOTATION_PROPERTIES_INK_ENABLED = "application.annotation.properties.ink.enabled"; public static final String PROPERTY_ANNOTATION_PROPERTIES_FREE_TEXT_ENABLED = "application.annotation.properties.freetext.enabled"; public static final String PROPERTY_ANNOTATION_PROPERTIES_TEXT_ENABLED = "application.annotation.properties.text.enabled"; public static final String PROPERTY_ANNOTATION_EDITING_MODE_ENABLED = "application.annotation.editing.mode.enabled"; // Individual controls for the annotation toolbar button commands public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_SELECTION = "application.toolbar.annotation.selection.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_DELETE = "application.toolbar.annotation.delete.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_HIGHLIGHT = "application.toolbar.annotation.highlight.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_REDACTION = "application.toolbar.annotation.redaction" + ".enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_UNDERLINE = "application.toolbar.annotation.underline.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_STRIKE_OUT = "application.toolbar.annotation.strikeout.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_LINE = "application.toolbar.annotation.line.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_LINK = "application.toolbar.annotation.link.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_ARROW = "application.toolbar.annotation.arrow.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_RECTANGLE = "application.toolbar.annotation.rectangle.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_CIRCLE = "application.toolbar.annotation.circle.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_INK = "application.toolbar.annotation.ink.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_FREE_TEXT = "application.toolbar.annotation.freetext.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_TEXT = "application.toolbar.annotation.text.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_PERMISSION = "application.toolbar.annotation.permission.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_UTILITY = "application.toolbar.annotation.toolbar.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_BOOKMARK_UTILITY = "application.toolbar.bookmark.toolbar.enabled"; public static final String PROPERTY_SHOW_TOOLBAR_ANNOTATION_PREVIEW = "application.toolbar.annotation.preview.enabled"; // Individual control of the markup annotation context menu public static final String PROPERTY_SHOW_ANNOTATION_MARKUP_REPLY_TO = "application.annotation.show.markup.replyTo"; public static final String PROPERTY_SHOW_ANNOTATION_MARKUP_ADD_ANNOTATIONS = "application.annotation.show.markup.addAnnotations"; public static final String PROPERTY_SHOW_ANNOTATION_MARKUP_SET_STATUS = "application.annotation.show.markup.setStatus"; // highlight annotation default colour as defined by the last used colour for each type. public static final String PROPERTY_ANNOTATION_HIGHLIGHT_COLOR = "application.viewer.preference.annotation.highlight.color"; public static final String PROPERTY_ANNOTATION_HIGHLIGHT_OPACITY = "application.viewer.preference.annotation.highlight.opacity"; public static final String PROPERTY_ANNOTATION_STRIKE_OUT_COLOR = "application.viewer.preference.annotation.strikeout.color"; public static final String PROPERTY_ANNOTATION_STRIKE_OUT_OPACITY = "application.viewer.preference.annotation.strikeout.opacity"; public static final String PROPERTY_ANNOTATION_UNDERLINE_COLOR = "application.viewer.preference.annotation.underline.color"; public static final String PROPERTY_ANNOTATION_UNDERLINE_OPACITY = "application.viewer.preference.annotation.underline.opacity"; public static final String PROPERTY_ANNOTATION_SQUIGGLY_COLOR = "application.viewer.preference.annotation.squiggly.color"; public static final String PROPERTY_ANNOTATION_SQUIGGLY_OPACITY = "application.viewer.preference.annotation.squiggly.opacity"; public static final String PROPERTY_ANNOTATION_TEXT_COLOR = "application.viewer.preference.annotation.text.color"; public static final String PROPERTY_ANNOTATION_TEXT_OPACITY = "application.viewer.preference.annotation.text.opacity"; public static final String PROPERTY_ANNOTATION_TEXT_ICON = "application.viewer.preference.annotation.text.icon"; public static final String PROPERTY_ANNOTATION_INK_COLOR = "application.viewer.preference.annotation.ink.color"; public static final String PROPERTY_ANNOTATION_INK_OPACITY = "application.viewer.preference.annotation.ink.opacity"; // annotation types with stroke and fill colours public static final String PROPERTY_ANNOTATION_SQUARE_COLOR = "application.viewer.preference.annotation.square.color"; public static final String PROPERTY_ANNOTATION_SQUARE_FILL_COLOR = "application.viewer.preference.annotation.square.fill.color"; public static final String PROPERTY_ANNOTATION_SQUARE_OPACITY = "application.viewer.preference.annotation.square.fill.opacity"; public static final String PROPERTY_ANNOTATION_CIRCLE_COLOR = "application.viewer.preference.annotation.circle.color"; public static final String PROPERTY_ANNOTATION_CIRCLE_FILL_COLOR = "application.viewer.preference.annotation.circle.fill.color"; public static final String PROPERTY_ANNOTATION_CIRCLE_OPACITY = "application.viewer.preference.annotation.circle.fill.opacity"; public static final String PROPERTY_ANNOTATION_LINE_COLOR = "application.viewer.preference.annotation.line.color"; public static final String PROPERTY_ANNOTATION_LINE_FILL_COLOR = "application.viewer.preference.annotation.line.fill.color"; public static final String PROPERTY_ANNOTATION_LINE_OPACITY = "application.viewer.preference.annotation.line.fill.opcity"; public static final String PROPERTY_ANNOTATION_LINE_ARROW_COLOR = "application.viewer.preference.annotation.arrow.color"; public static final String PROPERTY_ANNOTATION_LINE_ARROW_FILL_COLOR = "application.viewer.preference.annotation.arrow.fill.color"; public static final String PROPERTY_ANNOTATION_LINE_ARROW_OPACITY = "application.viewer.preference.annotation.arrow.fill.opacity"; // free text, quite a lot of properties public static final String PROPERTY_ANNOTATION_FREE_TEXT_COLOR = "application.viewer.preference.annotation.freetext.color"; public static final String PROPERTY_ANNOTATION_FREE_TEXT_SIZE = "application.viewer.preference.annotation.freetext.size"; public static final String PROPERTY_ANNOTATION_FREE_TEXT_FONT = "application.viewer.preference.annotation.freetext.font"; public static final String PROPERTY_ANNOTATION_FREE_TEXT_OPACITY = "application.viewer.preference.annotation.freetext.opacity"; public static final String PROPERTY_ANNOTATION_FREE_TEXT_FILL_COLOR = "application.viewer.preference.annotation.freetext.fill.color"; public static final String PROPERTY_ANNOTATION_FREE_TEXT_BORDER_COLOR = "application.viewer.preference.annotation.freetext.border.color"; // we use the same recent colour list for all annotation types public static final String PROPERTY_ANNOTATION_RECENT_COLORS = "application.viewer.preference.annotation.color.recent"; // resent colour and labels, enabled automatically if there is more then one. public static final String PROPERTY_ANNOTATION_RECENT_COLOR_LABEL = "application.viewer.preference.annotation.recent.color.labels"; // store for recently opened files. public static final String PROPERTY_RECENTLY_OPENED_FILES = "application.viewer.preference.recent.files"; // store of sort, filter and quick colour markup annotation utility pane persisted values. public static final String PROPERTY_ANNOTATION_SORT_COLUMN = "application.viewer.utility.annotation.sort.column"; public static final String PROPERTY_ANNOTATION_FILTER_AUTHOR_COLUMN = "application.viewer.utility.annotation.filter.author.column"; public static final String PROPERTY_ANNOTATION_FILTER_VISIBILITY_COLUMN = "application.viewer.utility.annotation.filter.visibility.column"; public static final String PROPERTY_ANNOTATION_FILTER_TYPE_COLUMN = "application.viewer.utility.annotation.filter.type.column"; public static final String PROPERTY_ANNOTATION_FILTER_COLOR_COLUMN = "application.viewer.utility.annotation.filter.color.column"; public static final String PROPERTY_ANNOTATION_QUICK_COLOR = "application.viewer.utility.annotation.filter.quick.color"; // text search panel settings public static final String PROPERTY_SEARCH_PANEL_WHOLE_PAGE_ENABLED = "application.viewer.utility.search.whole.page.enabled"; public static final String PROPERTY_SEARCH_PANEL_REGEX_ENABLED = "application.viewer.utility.search.regex.enabled"; public static final String PROPERTY_SEARCH_PANEL_WHOLE_WORDS_ENABLED = "application.viewer.utility.search.whole.words.enabled"; public static final String PROPERTY_SEARCH_PANEL_CASE_SENSITIVE_ENABLED = "application.viewer.utility.search.case.sensitive.enabled"; public static final String PROPERTY_SEARCH_PANEL_CUMULATIVE_ENABLED = "application.viewer.utility.search.case.cumulative.enabled"; public static final String PROPERTY_SEARCH_PANEL_SEARCH_COMMENTS_ENABLED = "application.viewer.utility.search.comments.cumulative.enabled"; public static final String PROPERTY_SEARCH_PANEL_SEARCH_DEST_ENABLED = "application.viewer.utility.search.case.destinations.enabled"; public static final String PROPERTY_SEARCH_PANEL_SEARCH_TEXT_ENABLED = "application.viewer.utility.search.case.text.enabled"; public static final String PROPERTY_SEARCH_PANEL_SEARCH_FORMS_ENABLED = "application.viewer.utility.search.case.forms.enabled"; public static final String PROPERTY_SEARCH_PANEL_SEARCH_OUTLINES_ENABLED = "application.viewer.utility.search.case.outlines.enabled"; public static final String PROPERTY_SEARCH_PANEL_SHOW_PAGES_ENABLED = "application.viewer.utility.search.case.pages.enabled"; public static final String PROPERTY_QUICK_SEARCH_WHOLE_WORDS_ENABLED = "application.viewer.toolbar.search.whole.words.enabled"; public static final String PROPERTY_QUICK_SEARCH_CASE_SENSITIVE_ENABLED = "application.viewer.toolbar.search.case.sensitive.enabled"; public static final String PROPERTY_QUICK_SEARCH_SEARCH_COMMENTS_ENABLED = "application.viewer.toolbar.search.comments.cumulative.enabled"; // markup search panel settings public static final String PROPERTY_SEARCH_MARKUP_PANEL_REGEX_ENABLED = "application.viewer.utility.search.markup.regex.enabled"; public static final String PROPERTY_SEARCH_MARKUP_PANEL_CASE_SENSITIVE_ENABLED = "application.viewer.utility.search.markup.case.sensitive.enabled"; // annotation summary panel font size and name. public static final String PROPERTY_ANNOTATION_SUMMARY_FONT_NAME = "application.viewer.annotation.summary.font.name"; public static final String PROPERTY_ANNOTATION_SUMMARY_FONT_SIZE = "application.viewer.annotation.summary.font.size"; // stored state of last used public/private annotation flag. public static final String PROPERTY_ANNOTATION_LAST_USED_PUBLIC_FLAG = "application.viewer.annotation.public.flag"; private static ViewerPropertiesManager propertiesManager; // static store of properties which are persisted to backing store. private static Preferences preferences = Preferences.userNodeForPackage(ViewerPropertiesManager.class); // local properties, that aren't persisted and can override properties in the store if root accessor are used. private static Properties localProperties; // default properties file included int the viewer jar private static Properties defaultProps; private ViewerPropertiesManager() { } /** * Gets singleton instance of the the Properties manager instance. * * @return singleton instance. */ public static ViewerPropertiesManager getInstance() { if (propertiesManager == null) { propertiesManager = new ViewerPropertiesManager(); try { if (preferences.keys().length == 0) { // load default properties from viewer jar and assigned to defaultProps. setupDefaultProperties(); saveLocalProperties(); } else { updatePropertiesWithPreferences(); } } catch (BackingStoreException e) { logger.severe("Couldn't access backing store"); setupDefaultProperties(); } } return propertiesManager; } /** * Prints all the preferences and properties. Can be used to see discrepancies. */ public static void printAllProperties() { try { Arrays.stream(preferences.keys()).forEach(key -> { logger.fine(key + " -> " + preferences.get(key, "")); logger.fine(key + " -> " + propertiesManager.checkAndStoreStringProperty(key, "")); }); } catch (BackingStoreException e) { e.printStackTrace(); } } private static void updatePropertiesWithPreferences() throws BackingStoreException { defaultProps = new SortedProperties(); Arrays.stream(preferences.keys()).forEach(key -> defaultProps.setProperty(key, preferences.get(key, null))); localProperties = new SortedProperties(defaultProps); } /** * Gets the Preferences backing store for persisting static properties and settings. * * @return reference to the application preferences backing store. */ public Preferences getPreferences() { return preferences; } /** * Sets the local property over writing any previous value stored in the default properties file. * * @param propertyName name of property to write. * @param value property value. */ public void set(String propertyName, String value) { localProperties.setProperty(propertyName, value); preferences.put(propertyName, value); } /** * Removes the * */ public void remove(String propertyName) { localProperties.remove(propertyName); } /** * All of the properties that are stored in the local properties can be persisted to the backing store via this * method call. The local properties are stored via the checkAndStore*() method calls and should only be used * when configuring the viewer components functionality. Once these properties have been persisted they are * now sticky and will persist for all viewer instances. */ public static void saveLocalProperties() { Enumeration keys = localProperties.keys(); while (keys.hasMoreElements()) { String key = (String) keys.nextElement(); preferences.put(key, localProperties.getProperty(key)); } } /** * Sets the local double property over writing any previous value stored in the default properties file. * * @param propertyName name of property to write. * @param value property value. */ public void setDouble(String propertyName, double value) { localProperties.setProperty(propertyName, Double.toString(value)); preferences.putDouble(propertyName, value); } /** * Sets the local float property over writing any previous value stored in the default properties file. * * @param propertyName name of property to write. * @param value property value. */ public void setFloat(String propertyName, float value) { localProperties.setProperty(propertyName, Float.toString(value)); preferences.putFloat(propertyName, value); } /** * Sets the local integer property over writing any previous value stored in the default properties file. * * @param propertyName name of property to write. * @param value property value. */ public void setInt(String propertyName, int value) { localProperties.setProperty(propertyName, Integer.toString(value)); preferences.putInt(propertyName, value); } /** * Sets the local boolean property over writing any previous value stored in the default properties file. * * @param propertyName name of property to write. * @param value property value. */ public void setBoolean(String propertyName, boolean value) { localProperties.setProperty(propertyName, Boolean.toString(value)); preferences.putBoolean(propertyName, value); } /** * Method to check the value of a string property * This is meant to be used for configuration via the properties file * After the property has been checked, it will be stored back into the Properties * object (using a default value if none was found) * * @param propertyName to check for * @param defaultVal to default to if no value is found on a property * @return String value for the propertyName or defaultVal if none exists. */ public String checkAndStoreStringProperty(String propertyName, String defaultVal) { // Get the desired property, defaulting to the defaultVal parameter String returnValue = localProperties.getProperty(propertyName, defaultVal); // Set the property back into the manager // This is necessary in the cases where a property didn't exist, but needs to be added to the file localProperties.setProperty(propertyName, returnValue); return returnValue; } public boolean checkAndStoreBooleanProperty(String propertyName) { return checkAndStoreBooleanProperty(propertyName, true); } /** * Method to check the value of a boolean property * This is meant to be used for configuration via the properties file * After the property has been checked, it will be stored back into the Properties * object (using a default value if none was found) * * @param propertyName to check for * @param defaultVal to default to if no value is found on a property * @return true if property is true, otherwise false */ public boolean checkAndStoreBooleanProperty(String propertyName, boolean defaultVal) { // Get the desired property, defaulting to the defaultVal parameter boolean returnValue = Boolean.parseBoolean( localProperties.getProperty(propertyName, Boolean.toString(defaultVal))); // Set the property back into the manager // This is necessary in the cases where a property didn't exist, but needs to be added to the file localProperties.setProperty(propertyName, String.valueOf(returnValue)); return returnValue; } public double checkAndStoreDoubleProperty(String propertyName) { return checkAndStoreDoubleProperty(propertyName, 1.0f); } /** * Method to check the value of a double property * This is meant to be used for configuration via the properties file * After the property has been checked, it will be stored back into the Properties * object (using a default value if none was found) * * @param propertyName to check for * @param defaultVal to default to if no value is found on a property * @return double property value */ public double checkAndStoreDoubleProperty(String propertyName, double defaultVal) { // Get the desired property, defaulting to the defaultVal parameter double returnValue = Double.parseDouble(localProperties.getProperty(propertyName, Double.toString(defaultVal))); // Set the property back into the manager // This is necessary in the cases where a property didn't exist, but needs to be added to the file localProperties.setProperty(propertyName, String.valueOf(returnValue)); return returnValue; } public int checkAndStoreIntProperty(String propertyName) { return checkAndStoreIntProperty(propertyName, 1); } /** * Method to check the value of an int property * This is meant to be used for configuration via the properties file * After the property has been checked, it will be stored back into the Properties * object (using a default value if none was found) * * @param propertyName to check for * @param defaultVal to default to if no value is found on a property * @return int value of property */ public int checkAndStoreIntProperty(String propertyName, int defaultVal) { // Get the desired property, defaulting to the defaultVal parameter int returnValue = Integer.parseInt(localProperties.getProperty(propertyName, Integer.toString(defaultVal))); // Set the property back into the manager // This is necessary in the cases where a property didn't exist, but needs to be added to the file localProperties.setProperty(propertyName, String.valueOf(returnValue)); return returnValue; } public float checkAndStoreFloatProperty(String propertyName) { return checkAndStoreFloatProperty(propertyName, 1); } /** * Method to check the value of an int property * This is meant to be used for configuration via the properties file * After the property has been checked, it will be stored back into the Properties * object (using a default value if none was found) * * @param propertyName to check for * @param defaultVal to default to if no value is found on a property * @return int value of property */ public float checkAndStoreFloatProperty(String propertyName, float defaultVal) { // Get the desired property, defaulting to the defaultVal parameter float returnValue = Float.parseFloat(localProperties.getProperty(propertyName, Float.toString(defaultVal))); // Set the property back into the manager // This is necessary in the cases where a property didn't exist, but needs to be added to the file localProperties.setProperty(propertyName, String.valueOf(returnValue)); return returnValue; } /** * Method to check the value of a comma separate list of floats property * For example we will convert "0.4f, 0.5f, 0.6f" to a size 3 array with the values as floats * This is meant to be used for configuration via the properties file * After the property has been checked, it will be stored back into the Properties * object (using a default value if none was found) * * @param propertyName to check for * @param defaultVal to default to if no value is found on a property * @return array of floats from the property */ public float[] checkAndStoreFloatArrayProperty(String propertyName, float[] defaultVal) { // Get the desired property, defaulting to the defaultVal parameter String propertyString = localProperties.getProperty(propertyName); float[] toReturn = defaultVal; try { // Ensure we have a property string to parse // Then we'll convert the comma separated property to a list of floats if ((propertyString != null) && (propertyString.trim().length() > 0)) { String[] split = propertyString.split(","); toReturn = new float[split.length]; for (int i = 0; i < split.length; i++) { try { toReturn[i] = Float.parseFloat(split[i]); } catch (NumberFormatException failedValue) { /* ignore as we'll just automatically put a '0' in the invalid space */ } } } // Otherwise convert the defaultVal into a comma separated list // This is done so it can be stored back into the properties file else { StringBuilder commaBuffer = new StringBuilder(defaultVal.length * 2); for (int i = 0; i < defaultVal.length; i++) { commaBuffer.append(defaultVal[i]); // Check whether we need a comma if ((i + 1) < defaultVal.length) { commaBuffer.append(","); } } // Set the property back into the manager // This is necessary in the cases where a property didn't exist, but needs to be added to the file localProperties.setProperty(propertyName, commaBuffer.toString()); } } catch (Exception failedProperty) { /* ignore on failure as we'll just return defaultVal */ } return toReturn; } /** * Allows users to set the default look and feel of the * * @param propertyName look and feel class and package name. * @param defaultValue default value * @param messageBundle message bundle. * @return class name used to set the look and feel */ public String getLookAndFeel(String propertyName, String defaultValue, ResourceBundle messageBundle) { String value = preferences.get(propertyName, null); if (value != null) { String result = Parse.parseLookAndFeel(value, messageBundle); if (result != null) { return result; } preferences.remove(propertyName); } if (defaultProps != null) { value = defaultProps.getProperty(propertyName); if (value != null) { String result = Parse.parseLookAndFeel(value, null); if (result != null) { return result; } preferences.remove(propertyName); Resources.showMessageDialog(null, JOptionPane.ERROR_MESSAGE, messageBundle, "manager.properties.title", "manager.properties.lafError", value); } } return defaultValue; } /** * This utility method removes all entries in the preferences backing store. This method does not remove the * preferences node and thus the font manager font cache is unaffected. */ public void clearPreferences() { try { String[] keys = preferences.keys(); for (String key : keys) { preferences.remove(key); } } catch (Exception ex) { // log the error if (logger.isLoggable(Level.WARNING)) { logger.log(Level.WARNING, "Error clearing preferences cache", ex); } } } /** * Reads the properties file that ships with the viewer jar and stores the default properties in the preferences * backing store. This is only done on first launch. */ private static void setupDefaultProperties() { // load the default values every time into our local store. try (InputStream in = getResourceAsStream(DEFAULT_PROP_FILE_PATH, DEFAULT_PROP_FILE)) { defaultProps = new SortedProperties(); if (in != null) { defaultProps.load(in); // we only set the default preferences on first load. if (preferences.get(PROPERTY_DEFAULT_FILE_PATH, null) == null) { Enumeration keys = defaultProps.keys(); while (keys.hasMoreElements()) { String key = (String) keys.nextElement(); preferences.put(key, defaultProps.getProperty(key)); } } } else if (logger.isLoggable(Level.FINER)) { logger.finer("Default properties file could not be found on the class path. "); } } catch (IOException ex) { // log the error if (logger.isLoggable(Level.WARNING)) { logger.log(Level.WARNING, "Error loading default properties cache", ex); } } // copy over the default properties. localProperties = new SortedProperties(defaultProps); } public static boolean importPreferences(File xml) { try (InputStream in = new BufferedInputStream(new FileInputStream(xml))) { Preferences.importPreferences(in); preferences = Preferences.userNodeForPackage(ViewerPropertiesManager.class); updatePropertiesWithPreferences(); return true; } catch (IOException | InvalidPreferencesFormatException | BackingStoreException e) { logger.warning("Couldn't load file " + xml.getAbsolutePath()); return false; } } public static boolean exportPreferences(File xml) { try (OutputStream out = new BufferedOutputStream(new FileOutputStream(xml))) { preferences.exportSubtree(out); return true; } catch (BackingStoreException | IOException e) { logger.warning("Couldn't write to file " + xml.getAbsolutePath()); return false; } } public static boolean importProperties(File file) { try (InputStream in = new BufferedInputStream(new FileInputStream(file))) { localProperties.load(in); saveLocalProperties(); return true; } catch (IOException e) { logger.warning("Couldn't load file " + file.getAbsolutePath()); return false; } } public static boolean exportProperties(File file) { try (OutputStream out = new BufferedOutputStream(new FileOutputStream(file))) { localProperties.store(out, "IcePDF properties file"); return true; } catch (IOException e) { logger.warning("Couldn't write to file " + file.getAbsolutePath()); return false; } } private static InputStream getResourceAsStream(String prefix, String resourcePath) { int colon = resourcePath.indexOf(':'); if (colon >= 0) { if (resourcePath.lastIndexOf(colon - 1, '/') < 0) { try { return (new URL(resourcePath)).openStream(); } catch (IOException e) { // eat the exception } return null; } } resourcePath = makeResPath(prefix, resourcePath); ClassLoader cl = prefix.getClass().getClassLoader(); if (cl != null) { InputStream result = cl.getResourceAsStream(resourcePath); if (result != null) { return result; } } return ClassLoader.getSystemResourceAsStream(resourcePath); } private static String makeResPath(String prefix, String base_name) { if (base_name.length() != 0 && base_name.charAt(0) == '/') { return base_name.substring(1); } else if (prefix == null) { return base_name; } else { return prefix + base_name; } } }