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

org.icepdf.ri.common.utility.search.SearchPanel Maven / Gradle / Ivy

/*
 * Copyright 2006-2017 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.common.utility.search;

import org.icepdf.core.pobjects.Document;
import org.icepdf.core.pobjects.graphics.text.LineText;
import org.icepdf.core.pobjects.graphics.text.WordText;
import org.icepdf.ri.common.SwingController;
import org.icepdf.ri.common.views.DocumentViewModelImpl;
import org.icepdf.ri.images.Images;
import org.icepdf.ri.util.SearchTextTask;

import javax.swing.*;
import javax.swing.event.TreeSelectionEvent;
import javax.swing.event.TreeSelectionListener;
import javax.swing.tree.*;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.text.ChoiceFormat;
import java.text.Format;
import java.text.MessageFormat;
import java.util.List;
import java.util.ResourceBundle;

/**
 * This class is the GUI component for the SearchTextTask.  This panel can be
 * added to a utility panel.  The GUI allows users to
 * type in a search string and displays a JList of all the pages that have results.
 * Each list item can be selected, and when selected, the viewer will show the
 * corresponding page.
 *
 * @since 1.1
 */
@SuppressWarnings("serial")
public class SearchPanel extends JPanel implements ActionListener,
        TreeSelectionListener {

    // markup for search context.
    private static final String HTML_TAG_START = "";
    private static final String HTML_TAG_END = "";
    private static final String BOLD_TAG_START = "";
    private static final String BOLD_TAG_END = "";

    // layouts constraint
    private GridBagConstraints constraints;
    // input for a search pattern
    private JTextField searchTextField;
    // pointer to document which will be searched
    private Document document;
    private SwingController controller;

    // tree view of the groups and panels
    //private ResultsTree resultsTree;

    // list box to hold search results
    private JTree tree;
    private DefaultMutableTreeNode rootTreeNode;
    private DefaultTreeModel treeModel;
    // search start button
    private JButton searchButton;
    // clear search
    private JButton clearSearchButton;
    // search option check boxes.
    private JCheckBox caseSensitiveCheckbox;
    private JCheckBox wholeWordCheckbox;
    private JCheckBox cumulativeCheckbox;
    private JCheckBox showPagesCheckbox;
    // page index of the last added node.
    private int lastNodePageIndex;

    // show progress of search
    protected JProgressBar progressBar;

    // task to complete in separate thread
    protected SearchTextTask searchTextTask;

    // status label for search
    protected JLabel findMessage;

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

    // refresh rate of gui elements
    private static final int ONE_SECOND = 1000;

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

    // message bundle for internationalization
    ResourceBundle messageBundle;
    MessageFormat searchResultMessageForm;

    /**
     * Create a new instance of SearchPanel.
     *
     * @param controller root SwingController
     */
    public SearchPanel(SwingController controller) {
        super(true);
        setFocusable(true);
        this.controller = controller;
        this.messageBundle = this.controller.getMessageBundle();
        searchResultMessageForm = setupSearchResultMessageForm();
        setGui();
        setDocument(controller.getDocument());
    }

    public void setDocument(Document doc) {
        // First have to stop any existing search
        if (timer != null)
            timer.stop();
        if (searchTextTask != null) {
            searchTextTask.stop();
            while (searchTextTask.isCurrentlySearching()) {
                try {
                    Thread.sleep(50L);
                } catch (Exception e) {
                    // intentional
                }
            }
        }

        document = doc;
        if (document != null && progressBar != null) {
            progressBar.setMaximum(document.getNumberOfPages());
        }
        if (searchTextField != null) {
            searchTextField.setText("");
        }
        if (searchButton != null) {
            searchButton.setText(messageBundle.getString(
                    "viewer.utilityPane.search.tab.title"));
        }
        if (rootTreeNode != null) {
            resetTree();
            // set title
            String docTitle = getDocumentTitle();
            rootTreeNode.setUserObject(docTitle);
            rootTreeNode.setAllowsChildren(true);
            tree.setRootVisible((docTitle != null));
        }
        if (findMessage != null) {
            findMessage.setText("");
            findMessage.setVisible(false);
        }
        if (progressBar != null) {
            progressBar.setVisible(false);
        }
        isSearching = false;
    }

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

        /**
         * Setup GUI objects
         */

        // 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);
        tree.addTreeSelectionListener(this);

        // set look and feel to match outline style
        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));

        // search Label
        JLabel searchLabel = new JLabel(messageBundle.getString(
                "viewer.utilityPane.search.searchText.label"));

        // search input field
        searchTextField = new JTextField("", 15);
        searchTextField.addActionListener(this);

        // setup search progress bar
        progressBar = new JProgressBar(0, 1);
        progressBar.setValue(0);
        progressBar.setVisible(false);
        findMessage = new JLabel(messageBundle.getString("viewer.utilityPane.search.searching.msg"));
        findMessage.setVisible(false);
        timer = new Timer(ONE_SECOND, new TimerListener());

        // setup search button
        searchButton = new JButton(messageBundle.getString(
                "viewer.utilityPane.search.searchButton.label"));
        searchButton.addActionListener(this);

        // clear search button
        clearSearchButton = new JButton(messageBundle.getString(
                "viewer.utilityPane.search.clearSearchButton.label"));
        clearSearchButton.addActionListener(this);

        // search options check boxes.
        wholeWordCheckbox = new JCheckBox(messageBundle.getString(
                "viewer.utilityPane.search.wholeWordCheckbox.label"));
        caseSensitiveCheckbox = new JCheckBox(messageBundle.getString(
                "viewer.utilityPane.search.caseSenstiveCheckbox.label"));
        cumulativeCheckbox = new JCheckBox(messageBundle.getString(
                "viewer.utilityPane.search.cumlitiveCheckbox.label"));
        showPagesCheckbox = new JCheckBox(messageBundle.getString(
                "viewer.utilityPane.search.showPagesCheckbox.label"), true);
        showPagesCheckbox.addActionListener(new ActionListener() {
            public void actionPerformed(ActionEvent event) {
                if (event.getSource() != null) {
                    // Determine if the user just selected or deselected the Show Pages checkbox
                    // If selected we'll want to combine all the leaf results into page nodes containing a series of results
                    // Otherwise we'll want to explode the parent/node page folders into basic leafs showing the results
                    if (((JCheckBox) event.getSource()).isSelected()) {
                        if ((rootTreeNode != null) && (rootTreeNode.getChildCount() > 0)) {
                            DefaultMutableTreeNode currentChild; // the current node we're handling
                            DefaultMutableTreeNode storedChildParent = null; // the newest page node we're adding to 
                            int newPageNumber; // page number of the current result node
                            int storedPageNumber = -1; // the page number of the node we're adding to                                  
                            int storedResultCount = 0; // the count of results that are on the storedPageNumber
                            Object[] messageArguments; // arguments used for formatting the labels

                            // Loop through the results tree
                            for (int i = 0; i < rootTreeNode.getChildCount(); i++) {
                                currentChild = (DefaultMutableTreeNode) rootTreeNode.getChildAt(i);

                                // Ensure we have a FindEntry object
                                if (currentChild.getUserObject() instanceof FindEntry) {
                                    newPageNumber = ((FindEntry) currentChild.getUserObject()).getPageNumber();

                                    // Check if the page number for the current node matches the stored number
                                    // If it does we will want to add the node to the existing page node,
                                    //  otherwise we'll want to create a new page node and start adding to that
                                    if (storedPageNumber == newPageNumber) {
                                        storedResultCount++;

                                        if (storedChildParent != null) {
                                            // Remove the old parentless child from the tree
                                            treeModel.removeNodeFromParent(currentChild);

                                            // Add the child back to the new page node
                                            storedChildParent.add(currentChild);
                                            currentChild.setParent(storedChildParent);

                                            // Reduce the loop count by one since we moved a node from the root to a page node
                                            i--;
                                        }
                                    } else {
                                        // Update the label of the page node, so that the result count is correct
                                        if (storedChildParent != null) {
                                            messageArguments = new Object[]{
                                                    String.valueOf(storedPageNumber + 1),
                                                    storedResultCount, storedResultCount};
                                            storedChildParent.setUserObject(
                                                    new FindEntry(searchResultMessageForm.format(messageArguments),
                                                            storedPageNumber));
                                        }

                                        // Reset the stored variables
                                        storedPageNumber = newPageNumber;
                                        storedResultCount = 1;

                                        treeModel.removeNodeFromParent(currentChild);

                                        // Create a new page node and move the current leaf to it
                                        messageArguments = new Object[]{
                                                String.valueOf(storedPageNumber + 1),
                                                storedResultCount, storedResultCount};
                                        storedChildParent = new DefaultMutableTreeNode(
                                                new FindEntry(searchResultMessageForm.format(messageArguments),
                                                        storedPageNumber),
                                                true);
                                        storedChildParent.add(currentChild);
                                        currentChild.setParent(storedChildParent);

                                        // Put the new page node into the overall tree
                                        treeModel.insertNodeInto(storedChildParent, rootTreeNode, i);
                                    }
                                }
                            }
                        }
                    } else {
                        if ((rootTreeNode != null) && (rootTreeNode.getChildCount() > 0)) {
                            // Now add the children back into the tree, this time without parent nodes
                            DefaultMutableTreeNode currentChild;
                            int rootChildCount = rootTreeNode.getChildCount();

                            // Loop through all children page nodes and explode the children out into leafs under the root node
                            // Then we'll remove the parent nodes so we're just left with leafs under the root
                            for (int i = 0; i < rootChildCount; i++) {
                                currentChild = (DefaultMutableTreeNode) rootTreeNode.getChildAt(0);

                                if (currentChild.getChildCount() > 0) {
                                    // Get any subchildren and reinsert them as plain leafs on the root
                                    // We need to wrap the user object in a new mutable tree node to stop any conflicts with parent indexes
                                    for (int j = 0; j < currentChild.getChildCount(); j++) {
                                        treeModel.insertNodeInto(
                                                new DefaultMutableTreeNode(
                                                        ((DefaultMutableTreeNode) currentChild.getChildAt(j)).getUserObject(),
                                                        false),
                                                rootTreeNode, rootTreeNode.getChildCount());
                                    }
                                }
                                treeModel.removeNodeFromParent(currentChild);
                            }
                        }
                    }
                }
            }
        });

        /**
         * Build search GUI
         */

        GridBagLayout layout = new GridBagLayout();
        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, 5, 1, 5);

        // content Panel
        JPanel searchPanel = new JPanel(layout);
        this.setLayout(new BorderLayout());
        this.add(new JScrollPane(searchPanel,
                JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED,
                JScrollPane.HORIZONTAL_SCROLLBAR_AS_NEEDED), BorderLayout.CENTER);

        // add the search label
        addGB(searchPanel, searchLabel, 0, 0, 2, 1);

        // add the search input field
        constraints.insets = new Insets(1, 1, 1, 5);
        constraints.weightx = 1.0;
        constraints.fill = GridBagConstraints.HORIZONTAL;
        addGB(searchPanel, searchTextField, 0, 1, 2, 1);

        // add start/stop search button
        constraints.insets = new Insets(1, 1, 1, 5);
        constraints.weightx = 1.0;
        constraints.fill = GridBagConstraints.EAST;
        addGB(searchPanel, searchButton, 0, 2, 1, 1);

        // add clear search button
        constraints.insets = new Insets(1, 1, 1, 5);
        constraints.weightx = 0;
        constraints.fill = GridBagConstraints.REMAINDER;
        addGB(searchPanel, clearSearchButton, 1, 2, 1, 1);

        // add case sensitive button
        constraints.insets = new Insets(5, 1, 1, 5);
        constraints.weightx = 1.0;
        constraints.fill = GridBagConstraints.HORIZONTAL;
        addGB(searchPanel, caseSensitiveCheckbox, 0, 3, 2, 1);

        // add whole word checkbox
        constraints.insets = new Insets(1, 1, 1, 5);
        constraints.fill = GridBagConstraints.HORIZONTAL;
        addGB(searchPanel, wholeWordCheckbox, 0, 4, 2, 1);

        // add cumulative checkbox
        constraints.insets = new Insets(1, 1, 1, 5);
        constraints.fill = GridBagConstraints.HORIZONTAL;
        addGB(searchPanel, cumulativeCheckbox, 0, 5, 2, 1);

        // add show pages checkbox
        constraints.insets = new Insets(1, 1, 1, 5);
        constraints.fill = GridBagConstraints.HORIZONTAL;
        addGB(searchPanel, showPagesCheckbox, 0, 6, 2, 1);

        // Add Results label
        constraints.insets = new Insets(10, 5, 1, 5);
        constraints.fill = GridBagConstraints.NONE;
        addGB(searchPanel, new JLabel(messageBundle.getString(
                        "viewer.utilityPane.search.results.label")),
                0, 7, 2, 1);

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

        // add find message
        constraints.insets = new Insets(1, 5, 1, 5);
        constraints.weighty = 0;
        constraints.fill = GridBagConstraints.NONE;
        constraints.anchor = GridBagConstraints.EAST;
        findMessage.setAlignmentX(JLabel.RIGHT_ALIGNMENT);
        addGB(searchPanel, findMessage, 0, 9, 2, 1);

        // add progress
        constraints.insets = new Insets(5, 5, 1, 5);
        constraints.fill = GridBagConstraints.HORIZONTAL;
        addGB(searchPanel, progressBar, 0, 10, 2, 1);
    }

    public void setVisible(boolean flag) {
        // try and get searchText focus
        super.setVisible(flag);
        if (this.isShowing()) {
            searchTextField.requestFocus(true);
        }
    }

    public void requestFocus() {
        super.requestFocus();
        // try and get searchText focus
        searchTextField.requestFocus();
    }

    public void dispose() {
        document = null;
        controller = null;
        searchTextTask = null;
        timer = null;
    }

    // Listen for selected tree items
    public void valueChanged(TreeSelectionEvent e) {
        // jump to the page stored in the JTree
        if (tree.getLastSelectedPathComponent() != null) {
            DefaultMutableTreeNode selectedNode = ((DefaultMutableTreeNode)
                    tree.getLastSelectedPathComponent());
            if (selectedNode.getUserObject() instanceof FindEntry) {
                // get the find entry and navigate to the page.
                FindEntry tmp = (FindEntry) selectedNode.getUserObject();
                if (controller != null) {
                    int oldTool = controller.getDocumentViewToolMode();
                    try {
                        controller.setDisplayTool(DocumentViewModelImpl.DISPLAY_TOOL_WAIT);
                        controller.showPage(tmp.getPageNumber());
                    } finally {
                        controller.setDisplayTool(oldTool);
                    }
                }
            }
        }
    }

    /**
     * Adds a new node item to the treeModel.
     *
     * @param title       display title of tree item
     * @param pageNumber  page number where the hit(s) occured
     * @param textResults list of LineText items that match
     * @param showPages   boolean to display or hide the page node
     */
    public void addFoundEntry(String title, int pageNumber,
                              List textResults,
                              boolean showPages) {
        // add the new results entry.
        if ((textResults != null) && (textResults.size() > 0)) {
            DefaultMutableTreeNode parentNode;
            // insert parent page number note.
            if ((showPages) &&
                    (lastNodePageIndex != pageNumber)) {
                parentNode = new DefaultMutableTreeNode(
                        new FindEntry(title, pageNumber), true);
                treeModel.insertNodeInto(parentNode, rootTreeNode,
                        rootTreeNode.getChildCount());
            } else {
                parentNode = rootTreeNode;
            }
            // add the hit entries.
            for (LineText currentText : textResults) {
                addObject(parentNode,
                        new DefaultMutableTreeNode(
                                new FindEntry(generateResultPreview(
                                        currentText.getWords()), pageNumber),
                                false), false);
            }

            // expand the root node, we only do this once.
            if (lastNodePageIndex == -1) {
                tree.expandPath(new TreePath(rootTreeNode));
            }

            lastNodePageIndex = pageNumber;

        }
    }

    /**
     * Utility for adding a tree node.
     *
     * @param parent          parent to add the node too.
     * @param childNode       node to add.
     * @param shouldBeVisible true indicates an auto scroll to bottom of page
     */
    private void addObject(DefaultMutableTreeNode parent,
                           DefaultMutableTreeNode childNode,
                           boolean shouldBeVisible) {
        if (parent == null) {
            parent = rootTreeNode;
        }
        //It is key to invoke this on the TreeModel, and NOT DefaultMutableTreeNode
        treeModel.insertNodeInto(childNode, parent,
                parent.getChildCount());
        //Make sure the user can see the lovely new node.
        if (shouldBeVisible) {
            tree.scrollPathToVisible(new TreePath(childNode.getPath()));
        }
    }

    /**
     * Generate the results preview label where the hit word is bolded using
     * html markup.
     *
     * @param allText text to unravel into a string.
     * @return styled html text.
     */
    private static String generateResultPreview(List allText) {
        StringBuilder toReturn = new StringBuilder(HTML_TAG_START);
        for (WordText currentText : allText) {
            if (currentText.isHighlighted()) {
                toReturn.append(BOLD_TAG_START);
                toReturn.append(currentText.getText());
                toReturn.append(BOLD_TAG_END);
            } else {
                toReturn.append(currentText.getText());
            }
        }
        toReturn.append(HTML_TAG_END);
        return toReturn.toString();
    }

    /**
     * Reset the tree for a new document or a new search.
     */
    protected void resetTree() {
        tree.setSelectionPath(null);
        rootTreeNode.removeAllChildren();
        treeModel.nodeStructureChanged(rootTreeNode);
        // rest node count
        lastNodePageIndex = -1;
    }

    /**
     * Utility for getting the doucment 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;
    }

    /**
     * Two main actions are handle here, search and clear search.
     *
     * @param event awt action event.
     */
    public void actionPerformed(ActionEvent event) {

        Object source = event.getSource();
        // Start/ stop a search
        if (searchTextField.getText().length() > 0 &&
                (source == searchTextField || source == searchButton)) {

            if (!timer.isRunning()) {
                // update gui components
                findMessage.setVisible(true);
                progressBar.setVisible(true);

                // clean the previous results and repaint the tree
                resetTree();

                // start a new search text task
                searchTextTask = new SearchTextTask(this,
                        controller,
                        searchTextField.getText(),
                        wholeWordCheckbox.isSelected(),
                        caseSensitiveCheckbox.isSelected(),
                        cumulativeCheckbox.isSelected(),
                        showPagesCheckbox.isSelected(),
                        false,
                        messageBundle);
                isSearching = true;

                // set state of search button
                searchButton.setText(messageBundle.getString(
                        "viewer.utilityPane.search.stopButton.label"));
                clearSearchButton.setEnabled(false);
                caseSensitiveCheckbox.setEnabled(false);
                wholeWordCheckbox.setEnabled(false);
                cumulativeCheckbox.setEnabled(false);
                showPagesCheckbox.setEnabled(false);

                // start the task and the timer
                searchTextTask.go();
                timer.start();
            } else {
                isSearching = false;
                clearSearchButton.setEnabled(true);
                caseSensitiveCheckbox.setEnabled(true);
                wholeWordCheckbox.setEnabled(true);
                cumulativeCheckbox.setEnabled(true);
                showPagesCheckbox.setEnabled(true);
            }
        } else if (source == clearSearchButton) {
            // clear input
            searchTextField.setText("");

            // clear the tree.
            resetTree();

            // reset high light states.
            controller.getDocumentSearchController().clearAllSearchHighlight();
            controller.getDocumentViewController().getViewContainer().repaint();
        }
    }

    /**
     * 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);
    }

    /**
     * Uitility for createing the searchable dialog message format.
     *
     * @return reuseable message format.
     */
    public MessageFormat setupSearchResultMessageForm() {
        MessageFormat messageForm =
                new MessageFormat(messageBundle.getString(
                        "viewer.utilityPane.search.result.msg"));
        double[] pageLimits = {0, 1, 2};
        String[] resultsStrings = {
                messageBundle.getString(
                        "viewer.utilityPane.search.result.moreFile.msg"),
                messageBundle.getString(
                        "viewer.utilityPane.search.result.oneFile.msg"),
                messageBundle.getString(
                        "viewer.utilityPane.search.result.moreFile.msg")
        };
        ChoiceFormat resultsChoiceForm = new ChoiceFormat(pageLimits,
                resultsStrings);

        Format[] formats = {null, resultsChoiceForm};
        messageForm.setFormats(formats);
        return messageForm;
    }

    /**
     * Uitility for createing the searching message format.
     *
     * @return reuseable message format.
     */
    public MessageFormat setupSearchingMessageForm() {
        // Build Internationalized plural phrase.
        MessageFormat messageForm =
                new MessageFormat(messageBundle.getString(
                        "viewer.utilityPane.search.searching1.msg"));
        double[] fileLimits = {0, 1, 2};
        String[] fileStrings = {
                messageBundle.getString(
                        "viewer.utilityPane.search.searching1.moreFile.msg"),
                messageBundle.getString(
                        "viewer.utilityPane.search.searching1.oneFile.msg"),
                messageBundle.getString(
                        "viewer.utilityPane.search.searching1.moreFile.msg"),
        };
        ChoiceFormat choiceForm = new ChoiceFormat(fileLimits,
                fileStrings);
        Format[] formats = {null, choiceForm, null};
        messageForm.setFormats(formats);

        return messageForm;
    }

    public MessageFormat setupSearchCompletionMessageForm() {
        MessageFormat messageForm =
                new MessageFormat(messageBundle.getString(
                        "viewer.utilityPane.search.progress.msg"));
        double[] pageLimits = {0, 1, 2};
        String[] pageStrings = {
                messageBundle.getString(
                        "viewer.utilityPane.search.progress.morePage.msg"),
                messageBundle.getString(
                        "viewer.utilityPane.search.progress.onePage.msg"),
                messageBundle.getString(
                        "viewer.utilityPane.search.progress.morePage.msg"),
        };
        ChoiceFormat pageChoiceForm = new ChoiceFormat(pageLimits,
                pageStrings);
        String[] resultsStrings = {
                messageBundle.getString(
                        "viewer.utilityPane.search.progress.moreMatch.msg"),
                messageBundle.getString(
                        "viewer.utilityPane.search.progress.oneMatch.msg"),
                messageBundle.getString(
                        "viewer.utilityPane.search.progress.moreMatch.msg"),
        };
        ChoiceFormat resultsChoiceForm = new ChoiceFormat(pageLimits,
                resultsStrings);

        Format[] formats = {null, pageChoiceForm, resultsChoiceForm};
        messageForm.setFormats(formats);
        return messageForm;
    }

    /**
     * The actionPerformed method in this class
     * is called each time the Timer "goes off".
     */
    class TimerListener implements ActionListener {
        public void actionPerformed(ActionEvent evt) {
            progressBar.setValue(searchTextTask.getCurrent());
            String s = searchTextTask.getMessage();
            if (s != null) {
                findMessage.setText(s);
            }
            // update the text when the search is completed
            if (searchTextTask.isDone() || !isSearching) {
                // update search status
                findMessage.setText(searchTextTask.getFinalMessage());
                timer.stop();
                searchTextTask.stop();
                // update buttons states. 
                searchButton.setText(messageBundle.getString("viewer.utilityPane.search.searchButton.label"));
                clearSearchButton.setEnabled(true);
                caseSensitiveCheckbox.setEnabled(true);
                wholeWordCheckbox.setEnabled(true);
                cumulativeCheckbox.setEnabled(true);
                showPagesCheckbox.setEnabled(true);

                // update progress bar then hide it.
                progressBar.setValue(progressBar.getMinimum());
                progressBar.setVisible(false);
            }
        }
    }

    /**
     * An Entry objects represents the found pages
     */
    @SuppressWarnings("serial")
    class FindEntry extends DefaultMutableTreeNode {

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

        // The destination to be displayed when this item is activated
        int pageNumber;

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

        /**
         * Get the page number.
         *
         * @return page number
         */
        public int getPageNumber() {
            return pageNumber;
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy