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

org.openide.awt.QuickSearch Maven / Gradle / Ivy

There is a newer version: RELEASE240
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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.openide.awt;

import java.awt.*;
import java.awt.event.*;
import java.lang.ref.WeakReference;
import java.util.function.Consumer;
import javax.swing.*;
import javax.swing.event.DocumentEvent;
import javax.swing.event.DocumentListener;
import javax.swing.event.PopupMenuEvent;
import javax.swing.event.PopupMenuListener;
import javax.swing.text.AbstractDocument;
import javax.swing.text.AttributeSet;
import javax.swing.text.BadLocationException;
import javax.swing.text.Document;
import javax.swing.text.DocumentFilter;
import org.netbeans.api.annotations.common.StaticResource;
import org.openide.util.ImageUtilities;
import org.openide.util.RequestProcessor;

/**
 * Quick search infrastructure for an arbitrary component.
 * When quick search is attached to a component, it listens on key events going
 * to the component and displays a quick search field.
 * 
 * @author Martin Entlicher
 * @since 7.43
 */
public class QuickSearch {
    
    @StaticResource
    private static final String ICON_FIND = "org/openide/awt/resources/quicksearch/find.png";      // NOI18N
    @StaticResource
    private static final String ICON_FIND_WITH_MENU = "org/openide/awt/resources/quicksearch/findMenu.png"; // NOI18N
    private static final Object CLIENT_PROPERTY_KEY = new Object();
    
    private final JComponent component;
    private final Object constraints;
    private final Callback callback;
    private final JMenu popupMenu;
    private final boolean asynchronous;
    private boolean enabled = true;
    private SearchTextField searchTextField;
    private KeyAdapter quickSearchKeyAdapter;
    private SearchFieldListener searchFieldListener;
    private JPanel searchPanel;
    private final RequestProcessor rp;
    private static enum QS_FIRE { UPDATE, NEXT, MAX }
    private AnimationTimer animationTimer;
    private boolean alwaysShown = false;
    private volatile boolean hasSearchText = false;
    
    private QuickSearch(JComponent component, Object constraints,
                        Callback callback, boolean asynchronous, JMenu popupMenu) {
        this.component = component;
        this.constraints = constraints;
        this.callback = callback;
        this.asynchronous = asynchronous;
        this.popupMenu = popupMenu;
        if (asynchronous) {
            rp = new RequestProcessor(QuickSearch.class);
        } else {
            rp = null;
        }
        setUpSearch();
    }
    
    /**
     * Attach quick search to a component with given constraints.
     * It listens on key events going to the component and displays a quick search
     * field.
     * 
     * @param component The component to attach to
     * @param constraints The constraints that are used to add the search field
     * to the component. It's passed to {@link JComponent#add(java.awt.Component, java.lang.Object)}
     * when adding the quick search UI to the component.
     * @param callback The call back implementation, which is notified from the
     * quick search field submissions.
     * @return An instance of QuickSearch class.
     */
    public static QuickSearch attach(JComponent component, Object constraints,
                                     Callback callback) {
        return attach(component, constraints, callback, false, null);
    }
    
    /**
     * Attach quick search to a component with given constraints.
     * It listens on key events going to the component and displays a quick search
     * field.
     * 
     * @param component The component to attach to
     * @param constraints The constraints that are used to add the search field
     * to the component. It's passed to {@link JComponent#add(java.awt.Component, java.lang.Object)}
     * when adding the quick search UI to the component.
     * @param callback The call back implementation, which is notified from the
     * quick search field submissions.
     * @return An instance of QuickSearch class.
     */
    public static QuickSearch attach(JComponent component, Object constraints,
                                     Callback callback, boolean asynchronous) {
        return attach(component, constraints, callback, asynchronous, null);
    }
    
    /**
     * Attach quick search to a component with given constraints.
     * It listens on key events going to the component and displays a quick search
     * field.
     * 
     * @param component The component to attach to
     * @param constraints The constraints that are used to add the search field
     * to the component. It's passed to {@link JComponent#add(java.awt.Component, java.lang.Object)}
     * when adding the quick search UI to the component.
     * @param callback The call back implementation, which is notified from the
     * quick search field submissions.
     * @param popupMenu A pop-up menu, that is displayed on the find icon, next to the search
     * field. This allows customization of the search criteria. The pop-up menu
     * is taken from {@link JMenu#getPopupMenu()}.
     * @return An instance of QuickSearch class.
     */
    public static QuickSearch attach(JComponent component, Object constraints,
                                     Callback callback, JMenu popupMenu) {
        return attach(component, constraints, callback, false, popupMenu);
    }
    /**
     * Attach quick search to a component with given constraints.
     * It listens on key events going to the component and displays a quick search
     * field.
     * 
     * @param component The component to attach to
     * @param constraints The constraints that are used to add the search field
     * to the component. It's passed to {@link JComponent#add(java.awt.Component, java.lang.Object)}
     * when adding the quick search UI to the component.
     * @param callback The call back implementation, which is notified from the
     * quick search field submissions.
     * @param asynchronous Set whether the quick search notifies the call back
     * asynchronously, or not.
     * By default, Callback is notified synchronously on EQ thread.
     * If true, three notification methods are called asynchronously
     * on a background thread. These are
     * {@link Callback#quickSearchUpdate(java.lang.String)},
     * {@link Callback#showNextSelection(javax.swing.text.Position.Bias)},
     * {@link Callback#findMaxPrefix(java.lang.String)}.
     * If false all methods are called synchronously on EQ thread.
     * @param popupMenu A pop-up menu, that is displayed on the find icon, next to the search
     * field. This allows customization of the search criteria. The pop-up menu
     * is taken from {@link JMenu#getPopupMenu()}.
     * @return An instance of QuickSearch class.
     */
    public static QuickSearch attach(JComponent component, Object constraints,
                                     Callback callback, boolean asynchronous, JMenu popupMenu) {
        Object qso = component.getClientProperty(CLIENT_PROPERTY_KEY);
        if (qso instanceof QuickSearch) {
            throw new IllegalStateException("A quick search is attached to this component already, detach it first."); // NOI18N
        } else {
            QuickSearch qs = new QuickSearch(component, constraints, callback, asynchronous, popupMenu);
            component.putClientProperty(CLIENT_PROPERTY_KEY, qs);
            return qs;
        }
    }
    
    /**
     * Detach the quick search from the component it was attached to.
     */
    public void detach() {
        setEnabled(false);
        component.putClientProperty(CLIENT_PROPERTY_KEY, null);
    }

    /**
     * Test whether the quick search field is always shown. 
     * This is false by default.
     * @return true when the search field is always shown,
     *                           false otherwise.
     * @since 7.49
     */
    public boolean isAlwaysShown() {
        return alwaysShown;
    }

    /**
     * Set whether the quick search field should always be shown.
     * @param alwaysShown true to always show the search field,
     *                           false otherwise.
     * @since 7.49
     */
    public void setAlwaysShown(boolean alwaysShown) {
        this.alwaysShown = alwaysShown;
        if(alwaysShown) {
            displaySearchField();
        } else {
            removeSearchField();
        }
    }
    
    /**
     * Test whether the quick search is enabled. This is true
     * by default.
     * @return true when the quick search is enabled,
     *         false otherwise.
     */
    public boolean isEnabled() {
        return enabled;
    }
    
    /**
     * Set the enabled state of the quick search.
     * This allows to activate/deactivate the quick search functionality.
     * @param enabled true to enable the quick search,
     *                false otherwise.
     */
    public void setEnabled(boolean enabled) {
        if (this.enabled == enabled) {
            return ;
        }
        this.enabled = enabled;
        if (enabled) {
            component.addKeyListener(quickSearchKeyAdapter);
        } else {
            removeSearchField();
            component.removeKeyListener(quickSearchKeyAdapter);
        }
    }
    
    /**
     * Process this key event in addition to the key events obtained from the
     * component we're attached to.
     * @param ke a key event to process.
     */
    public void processKeyEvent(KeyEvent ke) {
        if (!isEnabled()) {
            return ;
        }
        if (searchPanel != null) {
            if (!isKeyEventInSearchFieldIgnored(ke)) {
                searchTextField.setCaretPosition(searchTextField.getText().length());
                searchTextField.processKeyEvent(ke);
            }
        } else {
            switch(ke.getID()) {
                case KeyEvent.KEY_PRESSED:
                    quickSearchKeyAdapter.keyPressed(ke);
                    break;
                case KeyEvent.KEY_RELEASED:
                    quickSearchKeyAdapter.keyReleased(ke);
                    break;
                case KeyEvent.KEY_TYPED:
                    quickSearchKeyAdapter.keyTyped(ke);
                    break;
            }
        }
    }
    
    private boolean isKeyEventInSearchFieldIgnored(KeyEvent ke) {
        // Ignore DELETE key events unless the search field has focus
        if (ke.getKeyCode() == KeyEvent.VK_DELETE) {
            return !searchTextField.isFocusOwner();
        } else {
            return false;
        }
    }

    private void fireQuickSearchUpdate(String searchText) {
        if (asynchronous) {
            rp.post(new LazyFire(QS_FIRE.UPDATE, searchText));
        } else {
            callback.quickSearchUpdate(searchText);
        }
        hasSearchText = true;
    }
    
    private void fireShowNextSelection(boolean forward) {
        if (asynchronous) {
            rp.post(new LazyFire(QS_FIRE.NEXT, forward));
        } else {
            callback.showNextSelection(forward);
        }
    }
    
    private void findMaxPrefix(String prefix, Consumer newPrefixSetter) {
        if (asynchronous) {
            rp.post(new LazyFire(QS_FIRE.MAX, prefix, newPrefixSetter));
        } else {
            prefix = callback.findMaxPrefix(prefix);
            newPrefixSetter.accept(prefix);
        }
    }
    
    private void setUpSearch() {
        searchTextField = new SearchTextField();
        // create new key listeners
        quickSearchKeyAdapter = (
            new KeyAdapter() {
            @Override
                public void keyTyped(KeyEvent e) {
                    int modifiers = e.getModifiers();
                    int keyCode = e.getKeyCode();
                    char c = e.getKeyChar();

                    //#43617 - don't eat + and -
                    //#98634 - and all its duplicates dont't react to space
                    if ((c == '+') || (c == '-') || (c==' ')) return; // NOI18N

                    if (((modifiers > 0) && (modifiers != KeyEvent.SHIFT_MASK)) || e.isActionKey()) {
                        return;
                    }

                    if (Character.isISOControl(c) ||
                            (keyCode == KeyEvent.VK_SHIFT) ||
                            (keyCode == KeyEvent.VK_ESCAPE)) return;

                    displaySearchField();
                    
                    final KeyStroke stroke = KeyStroke.getKeyStrokeForEvent(e);
                    searchTextField.setText(String.valueOf(stroke.getKeyChar()));

                    e.consume();
                }
            }
        );
        if (isEnabled()) {
            component.addKeyListener(quickSearchKeyAdapter);
        }
        // Create a the "multi-event" listener for the text field. Instead of
        // adding separate instances of each needed listener, we're using a
        // class which implements them all. This approach is used in order 
        // to avoid the creation of 4 instances which takes some time
        searchFieldListener = new SearchFieldListener();
        searchTextField.addKeyListener(searchFieldListener);
        searchTextField.addFocusListener(searchFieldListener);
        Document searchDoc = searchTextField.getDocument();
        searchDoc.addDocumentListener(searchFieldListener);
        if (searchDoc instanceof AbstractDocument) {
            ((AbstractDocument) searchDoc).setDocumentFilter(searchFieldListener.new ReplaceFilter());
        }
        if(isAlwaysShown()) {
            displaySearchField();
        }
    }
    
    private void displaySearchField() {
        if (searchPanel != null || !isEnabled()) {
            return;
        }
        searchTextField.setOriginalFocusOwner();
        searchTextField.setFont(component.getFont());
        searchPanel = new SearchPanel(component, isAlwaysShown());
        final JLabel lbl;
        if (popupMenu != null) {
            lbl = new JLabel(org.openide.util.ImageUtilities.loadImageIcon(ICON_FIND_WITH_MENU, false));
            lbl.addMouseListener(new MouseAdapter() {
                @Override
                public void mousePressed(MouseEvent e) {
                    maybeShowPopup(e, lbl);
                }
            });
        } else {
            lbl = new JLabel(org.openide.util.ImageUtilities.loadImageIcon(ICON_FIND, false));
        }
        if (asynchronous) {
            animationTimer = new AnimationTimer(lbl, lbl.getIcon());
        } else {
            animationTimer = null;
        }
        searchPanel.setLayout(new BoxLayout(searchPanel, BoxLayout.X_AXIS));
        if ("Aqua".equals(UIManager.getLookAndFeel().getID())) { //NOI18N
            if (popupMenu != null) {
                final JPopupMenu dummy = new JPopupMenu();
                dummy.addPopupMenuListener(new PopupMenuListener() {
                    @Override
                    public void popupMenuWillBecomeVisible(PopupMenuEvent e) {
                        SwingUtilities.invokeLater(new Runnable() {
                            @Override
                            public void run() {
                                dummy.setVisible(false);
                            }
                        });
                    }

                    @Override
                    public void popupMenuWillBecomeInvisible(PopupMenuEvent e) {
                    }

                    @Override
                    public void popupMenuCanceled(PopupMenuEvent e) {
                    }
                });

                searchTextField.putClientProperty("JTextField.Search.FindPopup", dummy); //NOI18N
                searchTextField.putClientProperty("JTextField.Search.FindAction", new ActionListener() { //NOI18N
                    @Override
                    public void actionPerformed(ActionEvent e) {
                        maybeShowPopup(null, searchTextField);
                    }
                });
            }
        } else {
            searchPanel.add(lbl);
        }
        searchPanel.add(searchTextField);
        searchPanel.setBackground(component.getBackground());
        lbl.setLabelFor(searchTextField);
        searchTextField.setColumns(10);
        searchTextField.setMaximumSize(searchTextField.getPreferredSize());
        searchTextField.putClientProperty("JTextField.variant", "search"); //NOI18N
        lbl.setBorder(BorderFactory.createEmptyBorder(0, 0, 0, 5));
        if (constraints == null) {
            component.add(searchPanel);
        } else {
            component.add(searchPanel, constraints);
        }
        component.invalidate();
        component.revalidate();
        component.repaint();
        searchTextField.requestFocus();
        searchTextField.selectAll(); // Select an existing text for an easy rewrite
    }
    
    protected void maybeShowPopup(MouseEvent evt, Component comp) {
        if (evt != null && !SwingUtilities.isLeftMouseButton(evt)) {
            return;
        }
        JPopupMenu pm = popupMenu.getPopupMenu();
        pm.show(comp, 0, comp.getHeight() - 1);
        searchTextField.setText("");
        searchTextField.requestOriginalFocusOwner();
    }
    
    private void removeSearchField() {
        if (isAlwaysShown()) {
            return;
        }
        if (searchPanel == null) {
            return;
        }
        if (animationTimer != null) {
            animationTimer.stopProgressAnimation();
        }
        Component sp = searchPanel;
        searchPanel = null;
        component.remove(sp);
        component.invalidate();
        component.revalidate();
        component.repaint();
    }
    
    /** Accessed from test. */
    JTextField getSearchField() {
        return searchTextField;
    }
    
    /**
     * Utility method, that finds a greatest common prefix of two supplied
     * strings.
     * 
     * @param str1 The first string
     * @param str2 The second string
     * @param ignoreCase Whether to ignore case in the comparisons
     * @return The greatest common prefix of the two strings.
     */
    public static String findMaxPrefix(String str1, String str2, boolean ignoreCase) {
        int n1 = str1.length();
        int n2 = str2.length();
        int i = 0;
        if (ignoreCase) {
            for ( ; i < n1 && i < n2; i++) {
                char c1 = Character.toUpperCase(str1.charAt(i));
                char c2 = Character.toUpperCase(str2.charAt(i));
                if (c1 != c2) {
                    break;
                }
            }
        } else {
            for ( ; i < n1 && i < n2; i++) {
                char c1 = str1.charAt(i);
                char c2 = str2.charAt(i);
                if (c1 != c2) {
                    break;
                }
            }
        }
        return str1.substring(0, i);
    }
    
    private static final class AnimationTimer {
        
        private final JLabel jLabel;
        private final Icon findIcon;
        private final Timer animationTimer;
        
        public AnimationTimer(final JLabel jLabel, Icon findIcon) {
            this.jLabel = jLabel;
            this.findIcon = findIcon;
            animationTimer = new Timer(100, new ActionListener() {

                ImageIcon icons[];
                int index = 0;

                @Override
                public void actionPerformed(ActionEvent e) {
                    if (icons == null) {
                        icons = new ImageIcon[8];
                        for (int i = 0; i < 8; i++) {
                            icons[i] = ImageUtilities.loadImageIcon("org/openide/awt/resources/quicksearch/progress_" + i + ".png", false);  //NOI18N
                        }
                    }
                    jLabel.setBorder(javax.swing.BorderFactory.createEmptyBorder(1, 1, 1, 6));
                    jLabel.setIcon(icons[index]);
                    //mac os x
                    jLabel.repaint();

                    index = (index + 1) % 8;
                }
            });
        }
        
        public void startProgressAnimation() {
            if (animationTimer != null && !animationTimer.isRunning()) {
                animationTimer.start();
            }
        }

        public void stopProgressAnimation() {
            if (animationTimer != null && animationTimer.isRunning()) {
                animationTimer.stop();
                jLabel.setIcon(findIcon);
                jLabel.setBorder(javax.swing.BorderFactory.createEmptyBorder(1, 1, 1, 1));
            }
        }

    }

    private class LazyFire implements Runnable {
        
        private final QS_FIRE fire;
        //private final QuickSearchListener[] qsls;
        private final String searchText;
        private final boolean forward;
        private final Consumer newPrefixSetter;
        
        LazyFire(QS_FIRE fire, String searchText) {
            this(fire, searchText, true, null);
        }
        
        LazyFire(QS_FIRE fire, boolean forward) {
            this(fire, null, forward);
        }
        
        LazyFire(QS_FIRE fire, String searchText, boolean forward) {
            this(fire, searchText, forward, null);
        }
        
        LazyFire(QS_FIRE fire, String searchText,
                 Consumer newPrefixSetter) {
            this(fire, searchText, true, newPrefixSetter);
        }
        
        LazyFire(QS_FIRE fire, String searchText, boolean forward,
                 Consumer newPrefixSetter) {
            this.fire = fire;
            //this.qsls = qsls;
            this.searchText = searchText;
            this.forward = forward;
            this.newPrefixSetter = newPrefixSetter;
            animationTimer.startProgressAnimation();
        }

        @Override
        public void run() {
            try {
            switch (fire) {
                case UPDATE:    callback.quickSearchUpdate(searchText);//fireQuickSearchUpdate(qsls, searchText);
                                break;
                case NEXT:      callback.showNextSelection(forward);//fireShowNextSelection(qsls, forward);
                                break;
                case MAX:       String mp = callback.findMaxPrefix(searchText);//String mp = findMaxPrefix(qsls, searchText);
                                newPrefixSetter.accept(mp);
                                break;
            }
            } finally {
                animationTimer.stopProgressAnimation();
            }
        }
    }
    
    private static class SearchPanel extends JPanel {
        
        public static final boolean isAquaLaF =
                "Aqua".equals(UIManager.getLookAndFeel().getID()); //NOI18N
        private JComponent component;
        private boolean alwaysShown = false;
    
        public SearchPanel(JComponent component, boolean alwaysShown) {
            this.component = component;
            this.alwaysShown = alwaysShown;
            if (isAquaLaF) {
                setBorder(BorderFactory.createEmptyBorder(9,6,8,2));
            } else {
                setBorder(BorderFactory.createEmptyBorder(2,6,2,2));
            }
            setOpaque(true);
        }

        @Override
        protected void paintComponent(Graphics g) {
            if (isAquaLaF && g instanceof Graphics2D) {
                Graphics2D g2d = (Graphics2D) g;
                if (alwaysShown) {
                    g2d.setColor(component.getBackground());
                    g2d.fillRect(0, 0, getWidth(), getHeight());
                } else {
                    g2d.setPaint(new GradientPaint(0, 0, UIManager.getColor("NbExplorerView.quicksearch.background.top"), //NOI18N
                            0, getHeight(), UIManager.getColor("NbExplorerView.quicksearch.background.bottom")));//NOI18N
                    g2d.fillRect(0, 0, getWidth(), getHeight());
                    g2d.setColor(UIManager.getColor("NbExplorerView.quicksearch.border")); //NOI18N
                    g2d.drawLine(0, 0, getWidth(), 0);
                }
            } else {
                super.paintComponent(g);
            }
        }
    }

    /** searchTextField manages focus because it handles VK_ESCAPE key */
    private class SearchTextField extends JTextField {
        
        private WeakReference originalFocusOwner = new WeakReference(null);
        
        public SearchTextField() {
        }
        
        void setOriginalFocusOwner() {
            Component focusOwner = KeyboardFocusManager.getCurrentKeyboardFocusManager().getFocusOwner();
            if (focusOwner != null && component.isAncestorOf(focusOwner)) {
                originalFocusOwner = new WeakReference(focusOwner);
            } else {
                originalFocusOwner = new WeakReference(component);
            }
        }
        
        void requestOriginalFocusOwner() {
            SwingUtilities.invokeLater(
                new Runnable() {
                    //additional bugfix - do focus change later or removing
                    //the component while it's focused will cause focus to
                    //get transferred to the next component in the
                    //parent focusTraversalPolicy *after* our request
                    //focus completes, so focus goes into a black hole - Tim
                    @Override
                    public void run() {
                        Component fo = originalFocusOwner.get();
                        if (fo != null) {
                            fo.requestFocusInWindow();
                        }
                    }
                }
            );
        }
        
        @Override
        public boolean isManagingFocus() {
            return true;
        }

        @Override
        public void processKeyEvent(KeyEvent ke) {
            //override the default handling so that
            //the parent will never receive the escape key and
            //close a modal dialog
            if (ke.getKeyCode() == KeyEvent.VK_ESCAPE) {
                removeSearchField();
                ke.consume();
                searchFieldListener.ignoreEvents = true;
                try {
                    // Clear the text after ESC
                    setText("");                                                // NOI18N
                } finally {
                    searchFieldListener.ignoreEvents = false;
                }
                // bugfix #32909, reqest focus when search field is removed
                requestOriginalFocusOwner();
                //fireQuickSearchCanceled();
                callback.quickSearchCanceled();
                hasSearchText = false;
            } else {
                if (!hasSearchText) {
                    int keyCode = ke.getKeyCode();
                    if (keyCode == KeyEvent.VK_DOWN || keyCode == KeyEvent.VK_UP ||
                        keyCode == KeyEvent.VK_LEFT || keyCode == KeyEvent.VK_RIGHT ||
                        keyCode == KeyEvent.VK_TAB || keyCode == KeyEvent.VK_F3) {
                        // Ignore movement events when search text was not set
                        return ;
                    }
                }
                super.processKeyEvent(ke);
            }
        }
    };
    
    private class SearchFieldListener extends KeyAdapter implements DocumentListener, FocusListener {
        
        private boolean ignoreEvents;
        private boolean ignoreRemove;

        SearchFieldListener() {
        }

        @Override
        public void changedUpdate(DocumentEvent e) {
            if (ignoreEvents) return;
            searchForNode();
        }

        @Override
        public void insertUpdate(DocumentEvent e) {
            if (ignoreEvents) return;
            searchForNode();
        }

        @Override
        public void removeUpdate(DocumentEvent e) {
            if (ignoreEvents || ignoreRemove) return;
            searchForNode();
        }

        @Override
        public void keyPressed(KeyEvent e) {
            int keyCode = e.getKeyCode();

            if (keyCode == KeyEvent.VK_ESCAPE) {
                removeSearchField();
                searchTextField.requestOriginalFocusOwner();
                ignoreEvents = true;
                try {
                    // Clear the text after ESC
                    searchTextField.setText("");                                // NOI18N
                } finally {
                    ignoreEvents = false;
                }
                //fireQuickSearchCanceled();
                callback.quickSearchCanceled();
                hasSearchText = false;
                e.consume();
            } else if (keyCode == KeyEvent.VK_UP || (keyCode == KeyEvent.VK_F3 && e.isShiftDown())) {
                fireShowNextSelection(false);
                // Stop processing the event here. Otherwise it's dispatched
                // to the tree too (which scrolls)
                e.consume();
            } else if (keyCode == KeyEvent.VK_DOWN || keyCode == KeyEvent.VK_F3) {
                fireShowNextSelection(true);
                // Stop processing the event here. Otherwise it's dispatched
                // to the tree too (which scrolls)
                e.consume();
            } else if (keyCode == KeyEvent.VK_TAB) {
                findMaxPrefix(searchTextField.getText(), new Consumer() {
                    @Override
                    public void accept(final String maxPrefix) {
                        if (!SwingUtilities.isEventDispatchThread()) {
                            SwingUtilities.invokeLater(new Runnable() {
                                @Override
                                public void run() {
                                    accept(maxPrefix);
                                    searchTextField.transferFocus();
                                }
                            });
                            return ;
                        }
                        ignoreEvents = true;
                        try {
                            searchTextField.setText(maxPrefix);
                        } finally {
                            ignoreEvents = false;
                        }
                    }
                });

                e.consume();
            } else if (keyCode == KeyEvent.VK_ENTER) {
                removeSearchField();
                //fireQuickSearchConfirmed();
                callback.quickSearchConfirmed();

                component.requestFocusInWindow();
                e.consume();
            }
        }

        /** Searches for a node in the tree. */
        private void searchForNode() {
            String text = searchTextField.getText();
            if (text.isEmpty() && isAlwaysShown()) {
                callback.quickSearchCanceled();
                hasSearchText = false;
            } else {
                fireQuickSearchUpdate(text);
            }
        }

        @Override
        public void focusGained(FocusEvent e) {
            if (e.getSource() == searchTextField) {
                // make sure nothing is selected
                int n = searchTextField.getText().length();
                searchTextField.select(n, n);
            }
        }

        @Override
        public void focusLost(FocusEvent e) {
            if (e.isTemporary() || isAlwaysShown()) {
                return ;
            }
            Component oppositeComponent = e.getOppositeComponent();
            if (e.getSource() != searchTextField) {
                ((Component) e.getSource()).removeFocusListener(this);
            }
            if (oppositeComponent instanceof JMenuItem || oppositeComponent instanceof JPopupMenu) {
                oppositeComponent.addFocusListener(this);
                return ;
            }
            if (oppositeComponent == searchTextField) {
                return ;
            }
            if (searchPanel != null) {
                removeSearchField();
                //fireQuickSearchConfirmed();
                callback.quickSearchCanceled();
                hasSearchText = false;
            }
        }
        
        private class ReplaceFilter extends DocumentFilter {

            @Override
            public void replace(FilterBypass fb, int offset, int length, String text, AttributeSet attrs) throws BadLocationException {
                // Replace might do remove before insert. Suppress remove so that we get only one event with the final text
                if (text != null && !text.isEmpty()) {
                    // will insert
                    ignoreRemove = true;
                }
                try {
                    super.replace(fb, offset, length, text, attrs);
                } finally {
                    ignoreRemove = false;
                }
            }

        }

    }
    
    /**
     * Call back interface, that is notified with the submissions to the quick search field.
     * 
     * @author Martin Entlicher
     * @since 7.43
     */
    public static interface Callback {
        
        /**
         * Called with an updated search text.
         * When {@link #isAsynchronous()} is false
         * it's called in EQ thread, otherwise, it's called in a background thread.
         * The client should update the visual representation of the search results
         *  and then return.

* This method is called to initiate and update the search process. * @param searchText The new text to search for. */ void quickSearchUpdate(String searchText); /** * Called to select a next occurrence of the search result. * When {@link #isAsynchronous()} is false * it's called in EQ thread, otherwise, it's called in a background thread. * The client should update the visual representation of the search results * and then return.

* @param forward The direction of the next search result. * true for forward direction, * false for backward direction. */ void showNextSelection(boolean forward); /** * Find the maximum prefix among the search results, that starts with the provided string. * This method is called when user press TAB in the search field, to auto-complete * the maximum prefix. * When {@link #isAsynchronous()} is false * it's called in EQ thread, otherwise, it's called in a background thread. * Utility method {@link QuickSearch#findMaxPrefix(java.lang.String, java.lang.String, boolean)} * can be used by the implementation. * @param prefix The prefix to start with * @return The maximum prefix. */ String findMaxPrefix(String prefix); /** * Called when the quick search is confirmed by the user. * This method is called in EQ thread always. */ void quickSearchConfirmed(); /** * Called when the quick search is canceled by the user. * This method is called in EQ thread always. */ void quickSearchCanceled(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy