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

org.icepdf.ri.common.fonts.FontDialog Maven / Gradle / Ivy

package org.icepdf.ri.common.fonts;

import org.icepdf.core.pobjects.Document;
import org.icepdf.core.pobjects.fonts.Font;
import org.icepdf.ri.common.EscapeJDialog;
import org.icepdf.ri.common.SwingController;
import org.icepdf.ri.common.SwingWorker;
import org.icepdf.ri.images.Images;
import org.icepdf.ri.util.FontPropertiesManager;
import org.icepdf.ri.util.PropertiesManager;

import javax.swing.*;
import javax.swing.border.EtchedBorder;
import javax.swing.border.TitledBorder;
import javax.swing.tree.*;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.WindowEvent;
import java.awt.event.WindowListener;
import java.text.MessageFormat;
import java.util.ResourceBundle;


/**
 * This class is a reference implementation for displaying a PDF file's
 * font information.   The dialog will start a worker thread that will read all the document's font objects and
 * build a tree view of the all the fonts.  This font view is primarily for debug purposes to make it easier to track
 * font substitution results.  The dialog also provides an easy way to refresh the
 * "\.icesoft\icepdf-viewer\pdfviewerfontcache.properties' with out manually deleting the file and restarted the viewer.
 *
 * {@link org.icepdf.ri.common.fonts.FindFontsTask}
 * @since 6.1.3
 */
@SuppressWarnings("serial")
public class FontDialog extends EscapeJDialog implements ActionListener, WindowListener {

    // refresh rate of gui elements
    private static final int TIMER_REFRESH = 20;

    // pointer to document which will be searched
    private Document document;
    private SwingController controller;

    // list box to hold search results
    private JTree tree;
    private DefaultMutableTreeNode rootTreeNode;
    private DefaultTreeModel treeModel;
    // font look up start on creation, but ok button will kill the the process and close the dialog.
    private JButton okButton;
    // clear and rescan system for fonts and rewrite file.
    private JButton resetFontCacheButton;

    // task to complete in separate thread
    protected FindFontsTask findFontsTask;

    // status label for font search
    protected JLabel findMessage = new JLabel();

    // time class to manage gui updates
    protected Timer timer;

    // flag indicating if search is under way.
    private boolean isFindignFonts;

    // message bundle for internationalization
    private ResourceBundle messageBundle;
    private MessageFormat typeMessageForm;
    private MessageFormat encodingMessageForm;
    private MessageFormat actualTypeMessageForm;
    private MessageFormat actualFontMessageForm;

    // layouts constraint
    private GridBagConstraints constraints;

    /**
     * Create a new instance of SearchPanel.
     *
     * @param controller root SwingController
     */
    public FontDialog(Frame frame, SwingController controller, boolean isModal) {
        super(frame, isModal);
        setFocusable(true);
        this.controller = controller;
        this.messageBundle = this.controller.getMessageBundle();
        setGui();
        setDocument(controller.getDocument());
    }

    public void setDocument(Document doc) {
        // First have to stop any existing font searches,  this shouldn't happen...
        if (timer != null)
            timer.stop();
        if (findFontsTask != null) {
            findFontsTask.stop();
            while (findFontsTask.isCurrentlySearching()) {
                try {
                    Thread.sleep(50L);
                } catch (Exception e) {
                    // intentional
                }
            }
        }
        document = doc;
        if (rootTreeNode != null) {
            resetTree();
            // set title
            String docTitle = getDocumentTitle();
            rootTreeNode.setUserObject(docTitle);
            rootTreeNode.setAllowsChildren(true);
            tree.setRootVisible((docTitle != null));
        }
        // setup the new worker task.
        if (findMessage != null) {
            findMessage.setText("");
        }

        // start the task and the timer
        findFontsTask = new FindFontsTask(this, controller, messageBundle);
        findFontsTask.go();
        timer.start();
        isFindignFonts = true;
    }

    /**
     * Construct the GUI layout.
     */
    private void setGui() {

        typeMessageForm =
                new MessageFormat(messageBundle.getString("viewer.dialog.fonts.info.type.label"));
        encodingMessageForm =
                new MessageFormat(messageBundle.getString("viewer.dialog.fonts.info.encoding.label"));
        actualTypeMessageForm =
                new MessageFormat(messageBundle.getString("viewer.dialog.fonts.info.substitution.type.label"));
        actualFontMessageForm =
                new MessageFormat(messageBundle.getString("viewer.dialog.fonts.info.substitution.path.label"));

        setTitle(messageBundle.getString("viewer.dialog.fonts.title"));
        setResizable(true);

        addWindowListener(this);

        // build the supporting tree objects
        rootTreeNode = new DefaultMutableTreeNode();
        treeModel = new DefaultTreeModel(rootTreeNode);

        // build and customize the JTree
        tree = new JTree(treeModel);
        tree.setRootVisible(true);
        tree.setExpandsSelectedPaths(true);
        tree.setShowsRootHandles(true);
        tree.setScrollsOnExpand(true);
        tree.getSelectionModel().setSelectionMode(
                TreeSelectionModel.SINGLE_TREE_SELECTION);

        // set look and feel to match outline style, consider revising with font type icons.
        DefaultTreeCellRenderer renderer = new DefaultTreeCellRenderer();
        renderer.setOpenIcon(new ImageIcon(Images.get("page.gif")));
        renderer.setClosedIcon(new ImageIcon(Images.get("page.gif")));
        renderer.setLeafIcon(new ImageIcon(Images.get("page.gif")));
        tree.setCellRenderer(renderer);

        JScrollPane scrollPane = new JScrollPane(tree);
        scrollPane.setPreferredSize(new Dimension(150, 75));

        // setup refresh timer for the font scan progress.
        timer = new Timer(TIMER_REFRESH, new TimerListener());

        /**
         * Build search GUI
         */
        // content Panel
        JPanel fontPropertiesPanel = new JPanel(new GridBagLayout());
        fontPropertiesPanel.setBorder(new TitledBorder(new EtchedBorder(EtchedBorder.LOWERED),
                messageBundle.getString("viewer.dialog.fonts.border.label"),
                TitledBorder.LEFT,
                TitledBorder.DEFAULT_POSITION));
        this.setLayout(new BorderLayout(15, 15));
        this.add(fontPropertiesPanel, BorderLayout.CENTER);

        constraints = new GridBagConstraints();
        constraints.fill = GridBagConstraints.NONE;
        constraints.weightx = 1.0;
        constraints.weighty = 0;
        constraints.anchor = GridBagConstraints.NORTH;
        constraints.anchor = GridBagConstraints.WEST;
        constraints.insets = new Insets(10, 15, 1, 15);

        // add the lit to scroll pane
        constraints.fill = GridBagConstraints.BOTH;
        constraints.insets = new Insets(10, 15, 10, 15);
        constraints.weightx = 1.0;
        constraints.weighty = 1.0;
        addGB(fontPropertiesPanel, scrollPane, 0, 1, 2, 1);

        // add find message
        constraints.insets = new Insets(2, 10, 2, 10);
        constraints.weighty = 0;
        constraints.fill = GridBagConstraints.NONE;
        constraints.anchor = GridBagConstraints.WEST;
        findMessage.setAlignmentX(JLabel.RIGHT_ALIGNMENT);
        addGB(fontPropertiesPanel, findMessage, 0, 2, 2, 1);

        resetFontCacheButton = new JButton(messageBundle.getString("viewer.dialog.fonts.resetCache.label"));
        resetFontCacheButton.setToolTipText(messageBundle.getString("viewer.dialog.fonts.resetCache.tip"));
        resetFontCacheButton.addActionListener(this);
        constraints.insets = new Insets(2, 10, 2, 10);
        constraints.weighty = 0;
        constraints.fill = GridBagConstraints.NONE;
        constraints.anchor = GridBagConstraints.WEST;
        addGB(fontPropertiesPanel, resetFontCacheButton, 0, 3, 1, 1);

        okButton = new JButton(messageBundle.getString("viewer.button.ok.label"));
        okButton.addActionListener(this);
        constraints.insets = new Insets(2, 10, 2, 10);
        constraints.weighty = 0;
        constraints.fill = GridBagConstraints.NONE;
        constraints.anchor = GridBagConstraints.EAST;
        addGB(fontPropertiesPanel, okButton, 1, 4, 1, 1);

        setSize(640, 480);
        setLocationRelativeTo(getOwner());
    }

    /**
     * Adds a new node item to the treeModel.
     *
     * @param font font used to build node properties.
     */
    public void addFoundEntry(Font font) {
        DefaultMutableTreeNode fontNode = new DefaultMutableTreeNode(font.getBaseFont(), true);
        // add type sub node for type
        insertNode(font.getSubType(), typeMessageForm, fontNode);
        // add encoding.
        insertNode(font.getEncoding(), encodingMessageForm, fontNode);
        // add font substitution info.
        if (font.isFontSubstitution() && font.getFont() != null) {
            insertNode(font.getFont().getName(), actualTypeMessageForm, fontNode);
            insertNode(font.getFont().getSource(), actualFontMessageForm, fontNode);
        }
        addObject(rootTreeNode, fontNode);

        // expand the root node, we only do this once.
        tree.expandPath(new TreePath(rootTreeNode));
    }

    /**
     * Utility to aid in the creation of a new font properties node.
     * @param label label for node.
     * @param messageFormat message formatter
     * @param parent parent node.
     */
    private void insertNode(Object label, MessageFormat messageFormat, DefaultMutableTreeNode parent) {
        if (label != null) {
            Object[] messageArguments = {label.toString()};
            label = messageFormat.format(messageArguments);
            DefaultMutableTreeNode encodingNode = new DefaultMutableTreeNode(label, true);
            addObject(parent, encodingNode);
        }
    }

    /**
     * Utility for adding a tree node.
     *
     * @param parent    parent to add the node too.
     * @param childNode node to add.
     */
    private void addObject(DefaultMutableTreeNode parent,
                           DefaultMutableTreeNode childNode) {
        if (parent == null) {
            parent = rootTreeNode;
        }
        //It is key to invoke this on the TreeModel, and NOT DefaultMutableTreeNode
        treeModel.insertNodeInto(childNode, parent,
                parent.getChildCount());
    }

    // quick and dirty expand all.
    protected void expandAllNodes() {
        int rowCount = tree.getRowCount();
        int i = 0;
        while (i < rowCount) {
            tree.expandRow(i);
            i += 1;
            rowCount = tree.getRowCount();
        }
    }

    /**
     * Reset the tree, insuring it's empty
     */
    protected void resetTree() {
        tree.setSelectionPath(null);
        rootTreeNode.removeAllChildren();
        treeModel.nodeStructureChanged(rootTreeNode);
    }

    /**
     * Utility for getting the document title.
     *
     * @return document title, if non title then a simple search results
     * label is returned;
     */
    private String getDocumentTitle() {
        String documentTitle = null;
        if (document != null && document.getInfo() != null) {
            documentTitle = document.getInfo().getTitle();
        }
        if ((documentTitle == null) || (documentTitle.trim().length() == 0)) {
            return null;
        }
        return documentTitle;
    }

    /**
     * Insure the font search process is killed when the dialog is closed via the 'esc' key.
     */
    @Override
    public void dispose() {
        super.dispose();
        closeWindowOperations();
    }

    /**
     * Two main actions are handle here, search and clear search.
     *
     * @param event awt action event.
     */
    public void actionPerformed(ActionEvent event) {
        if (event.getSource() == okButton) {
            // clean up the timer and worker thread.
            closeWindowOperations();
            dispose();
        } else if (event.getSource() == resetFontCacheButton) {
            // reset the font properties cache.
            resetFontCacheButton.setEnabled(false);
            SwingWorker worker = new SwingWorker() {
                public Object construct() {
                    PropertiesManager properties = new PropertiesManager(
                            System.getProperties(),
                            ResourceBundle.getBundle(PropertiesManager.DEFAULT_MESSAGE_BUNDLE));
                    FontPropertiesManager fontPropertiesManager = new FontPropertiesManager(properties,
                            System.getProperties(), messageBundle);
                    fontPropertiesManager.clearProperties();
                    fontPropertiesManager.readDefaulFontPaths(null);
                    fontPropertiesManager.saveProperties();
                    resetFontCacheButton.setEnabled(true);

                    Runnable doSwingWork = new Runnable() {
                        public void run() {
                            resetFontCacheButton.setEnabled(true);
                        }
                    };
                    SwingUtilities.invokeLater(doSwingWork);
                    return null;
                }
            };
            worker.setThreadPriority(Thread.MIN_PRIORITY);
            worker.start();
        }
    }

    protected void closeWindowOperations() {
        // clean up the timer and worker thread.
        if (timer != null && timer.isRunning()) timer.stop();
        if (findFontsTask != null && findFontsTask.isCurrentlySearching()) findFontsTask.stop();
        setVisible(false);
    }


    /**
     * Gridbag constructor helper
     *
     * @param panel     parent adding component too.
     * @param component component to add to grid
     * @param x         row
     * @param y         col
     * @param rowSpan   rowspane of field
     * @param colSpan   colspane of field.
     */
    private void addGB(JPanel panel, Component component,
                       int x, int y,
                       int rowSpan, int colSpan) {
        constraints.gridx = x;
        constraints.gridy = y;
        constraints.gridwidth = rowSpan;
        constraints.gridheight = colSpan;
        panel.add(component, constraints);
    }


    public void windowOpened(WindowEvent e) {

    }

    public void windowClosing(WindowEvent e) {
        closeWindowOperations();
    }

    public void windowClosed(WindowEvent e) {
        closeWindowOperations();
    }

    public void windowIconified(WindowEvent e) {

    }

    public void windowDeiconified(WindowEvent e) {

    }

    public void windowActivated(WindowEvent e) {

    }

    public void windowDeactivated(WindowEvent e) {

    }

    /**
     * The actionPerformed method in this class
     * is called each time the Timer "goes off".
     */
    class TimerListener implements ActionListener {
        public void actionPerformed(ActionEvent evt) {
            String s = findFontsTask.getMessage();
            if (s != null) {
                findMessage.setText(s);
            }
            // update the text when the search is completed
            if (findFontsTask.isDone() || !isFindignFonts) {
                // update search status, blank it.
                findMessage.setText("");
                timer.stop();
                findFontsTask.stop();
            }
        }
    }

    /**
     * An Entry objects represents the found pages
     */

    @SuppressWarnings("serial")
    class FontEntry extends DefaultMutableTreeNode {

        // The text to be displayed on the screen for this item.
        String title;


        /**
         * Creates a new instance of a FindEntry.
         *
         * @param title      display title
         * @param pageNumber page number where the hit(s) occured
         */
        FontEntry(String title, int pageNumber) {
            super();
            this.title = title;
            setUserObject(title);
        }

    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy