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

com.jidesoft.swing.TextComponentSearchable Maven / Gradle / Ivy

There is a newer version: 3.6.18
Show newest version
/*
 * @(#)TextComponentSearchable.java 10/11/2005
 *
 * Copyright 2002 - 2005 JIDE Software Inc. All rights reserved.
 */
package com.jidesoft.swing;

import com.jidesoft.swing.event.SearchableEvent;

import javax.swing.*;
import javax.swing.event.DocumentEvent;
import javax.swing.event.DocumentListener;
import javax.swing.text.*;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.KeyEvent;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.util.HashMap;
import java.util.Iterator;

/**
 * TextComponentSearchable is an concrete implementation of {@link Searchable} that enables the search
 * function in JTextComponent. 

It's very simple to use it. Assuming you have a JTextComponent, all you need to do is * to call *

 * JTextComponent textComponent = ....;
 * TextComponentSearchable searchable = new TextComponentSearchable(textComponent);
 * 
* Now the JTextComponent will have the search function. *

* There is very little customization you need to do to ListSearchable. The only thing you might need is when the * element in the JTextComponent needs a special conversion to convert to string. If so, you can override * convertElementToString() to provide you own algorithm to do the conversion. *

 * JTextComponent textComponent = ....;
 * TextComponentSearchable searchable = new ListSearchable(textComponent) {
 *      protected String convertElementToString(Object object) {
 *          ...
 *      }
 * 

* protected boolean isActivateKey(KeyEvent e) { // change to a different activation key * return (e.getID() == KeyEvent.KEY_PRESSED && e.getKeyCode() == KeyEvent.VK_F && * (KeyEvent.CTRL_MASK & e.getModifiers()) != 0); * } * }; *

*

* Additional customization can be done on the base Searchable class such as background and foreground color, * keystrokes, case sensitivity. TextComponentSearchable also has a special attribute called highlightColor. You can * change it using {@link #setHighlightColor(java.awt.Color)}. *

* Due to the special case of JTextComponent, the searching doesn't support wild card '*' or '?' as in other * Searchables. The other difference is JTextComponent will keep the highlights after search popup hides. If you want to * hide the highlights, just press ESC again (the first ESC will hide popup; the second ESC will hide all highlights if * any). */ public class TextComponentSearchable extends Searchable implements DocumentListener, PropertyChangeListener { private Highlighter.HighlightPainter _highlightPainter; private static final Color DEFAULT_HIGHLIGHT_COLOR = new Color(204, 204, 255); private Color _highlightColor = null; private int _selectedIndex = -1; private HighlighCache _highlighCache; public TextComponentSearchable(JTextComponent textComponent) { super(textComponent); _highlighCache = new HighlighCache(); installHighlightsRemover(); setHighlightColor(DEFAULT_HIGHLIGHT_COLOR); } /** * Uninstalls the handler for ESC key to remove all highlights */ public void uninstallHighlightsRemover() { _component.unregisterKeyboardAction(KeyStroke.getKeyStroke(KeyEvent.VK_ESCAPE, 0)); } /** * Installs the handler for ESC key to remove all highlights */ public void installHighlightsRemover() { AbstractAction highlightRemover = new AbstractAction() { public void actionPerformed(ActionEvent e) { removeAllHighlights(); } }; _component.registerKeyboardAction(highlightRemover, KeyStroke.getKeyStroke(KeyEvent.VK_ESCAPE, 0), JComponent.WHEN_FOCUSED); } @Override public void installListeners() { super.installListeners(); if (_component instanceof JTextComponent) { ((JTextComponent) _component).getDocument().addDocumentListener(this); _component.addPropertyChangeListener("document", this); } } @Override public void uninstallListeners() { super.uninstallListeners(); if (_component instanceof JTextComponent) { ((JTextComponent) _component).getDocument().removeDocumentListener(this); _component.removePropertyChangeListener("document", this); } } @Override protected void setSelectedIndex(int index, boolean incremental) { if (_component instanceof JTextComponent) { if (index == -1) { removeAllHighlights(); _selectedIndex = -1; return; } if (!incremental) { removeAllHighlights(); } String text = getSearchingText(); try { addHighlight(index, text, incremental); } catch (BadLocationException e) { e.printStackTrace(); } } } /** * Adds highlight to text component at specified index and text. * * @param index the index of the text to be highlighted * @param text the text to be highlighted * @param incremental if this is an incremental adding highlight * @throws BadLocationException */ protected void addHighlight(int index, String text, boolean incremental) throws BadLocationException { if (_component instanceof JTextComponent) { JTextComponent textComponent = ((JTextComponent) _component); Object obj = textComponent.getHighlighter().addHighlight(index, index + text.length(), _highlightPainter); _highlighCache.addHighlight(obj); _selectedIndex = index; if (!incremental) { scrollTextVisible(textComponent, index, text.length()); } } } private void scrollTextVisible(JTextComponent textComponent, int index, int length) { // scroll highlight visible if (index != -1) { // Scroll the component if needed so that the composed text // becomes visible. try { Rectangle begin = textComponent.modelToView(index); if (begin == null) { return; } Rectangle end = textComponent.modelToView(index + length); if (end == null) { return; } Rectangle bounds = _component.getVisibleRect(); if (begin.x <= bounds.width) { // make sure if scroll back to the beginning as long as selected rect is visible begin.width = end.x; begin.x = 0; } else { begin.width = end.x - begin.x; } textComponent.scrollRectToVisible(begin); } catch (BadLocationException ble) { } } } /** * Removes all highlights from the text component. */ protected void removeAllHighlights() { if (_component instanceof JTextComponent) { Iterator itor = _highlighCache.getAllHighlights(); while (itor.hasNext()) { Object o = itor.next(); ((JTextComponent) _component).getHighlighter().removeHighlight(o); } _highlighCache.removeAllHighlights(); } } @Override protected int getSelectedIndex() { if (_component instanceof JTextComponent) { return _selectedIndex; } return 0; } @Override protected Object getElementAt(int index) { String text = getSearchingText(); if (text != null) { if (_component instanceof JTextComponent) { int endIndex = index + text.length(); int elementCount = getElementCount(); if (endIndex > elementCount) { endIndex = getElementCount(); } try { return ((JTextComponent) _component).getDocument().getText(index, endIndex - index + 1); } catch (BadLocationException e) { return null; } } } return ""; } @Override protected int getElementCount() { if (_component instanceof JTextComponent) { return ((JTextComponent) _component).getDocument().getLength(); } return 0; } /** * Converts the element in JTextComponent to string. The returned value will be the toString() of * whatever element that returned from list.getModel().getElementAt(i). * * @param object * @return the string representing the element in the JTextComponent. */ @Override protected String convertElementToString(Object object) { if (object != null) { return object.toString(); } else { return ""; } } public void propertyChange(PropertyChangeEvent evt) { if (isProcessModelChangeEvent()) { hidePopup(); _text = null; if (evt.getOldValue() instanceof Document) { ((Document) evt.getNewValue()).removeDocumentListener(this); } if (evt.getNewValue() instanceof Document) { ((Document) evt.getNewValue()).addDocumentListener(this); } fireSearchableEvent(new SearchableEvent(this, SearchableEvent.SEARCHABLE_MODEL_CHANGE)); } } public void insertUpdate(DocumentEvent e) { if (isProcessModelChangeEvent()) { hidePopup(); _text = null; fireSearchableEvent(new SearchableEvent(this, SearchableEvent.SEARCHABLE_MODEL_CHANGE)); } } public void removeUpdate(DocumentEvent e) { if (isProcessModelChangeEvent()) { hidePopup(); _text = null; fireSearchableEvent(new SearchableEvent(this, SearchableEvent.SEARCHABLE_MODEL_CHANGE)); } } public void changedUpdate(DocumentEvent e) { if (isProcessModelChangeEvent()) { hidePopup(); _text = null; fireSearchableEvent(new SearchableEvent(this, SearchableEvent.SEARCHABLE_MODEL_CHANGE)); } } @Override protected boolean isActivateKey(KeyEvent e) { if (_component instanceof JTextComponent && ((JTextComponent) _component).isEditable()) { return (e.getID() == KeyEvent.KEY_PRESSED && e.getKeyCode() == KeyEvent.VK_F && (KeyEvent.CTRL_MASK & e.getModifiers()) != 0); } else { return super.isActivateKey(e); } } /** * Gets the highlight color. * * @return the highlight color. */ public Color getHighlightColor() { if (_highlightColor != null) { return _highlightColor; } else { return DEFAULT_HIGHLIGHT_COLOR; } } /** * Changes the highlight color. * * @param highlightColor */ public void setHighlightColor(Color highlightColor) { _highlightColor = highlightColor; _highlightPainter = new DefaultHighlighter.DefaultHighlightPainter(_highlightColor); } @Override public int findLast(String s) { if (_component instanceof JTextComponent) { String text = getDocumentText(); if (isCaseSensitive()) { return text.lastIndexOf(s); } else { return text.toLowerCase().lastIndexOf(s.toLowerCase()); } } else { return super.findLast(s); } } private String _text = null; /** * Gets the text from Document. * * @return the text of this JTextComponent. It used Document to get the text. */ private String getDocumentText() { if (_text == null) { Document document = ((JTextComponent) _component).getDocument(); try { String text = document.getText(0, document.getLength()); _text = text; } catch (BadLocationException e) { return ""; } } return _text; } @Override public int findFirst(String s) { if (_component instanceof JTextComponent) { String text = getDocumentText(); if (isCaseSensitive()) { return text.indexOf(s); } else { return text.toLowerCase().indexOf(s.toLowerCase()); } } else { return super.findFirst(s); } } @Override public int findFromCursor(String s) { if (isReverseOrder()) { return reverseFindFromCursor(s); } if (_component instanceof JTextComponent) { String text = getDocumentText(); if (!isCaseSensitive()) { text = text.toLowerCase(); } String str = isCaseSensitive() ? s : s.toLowerCase(); int selectedIndex = (getCursor() != -1 ? getCursor() : getSelectedIndex()); if (selectedIndex < 0) selectedIndex = 0; int count = getElementCount(); if (count == 0) return s.length() > 0 ? -1 : 0; // find from cursor int found = text.indexOf(str, selectedIndex); // if not found, start over from the beginning if (found == -1) { found = text.indexOf(str, 0); if (found >= selectedIndex) { found = -1; } } return found; } else { return super.findFromCursor(s); } } @Override public int reverseFindFromCursor(String s) { if (!isReverseOrder()) { return findFromCursor(s); } if (_component instanceof JTextComponent) { String text = getDocumentText(); if (!isCaseSensitive()) { text = text.toLowerCase(); } String str = isCaseSensitive() ? s : s.toLowerCase(); int selectedIndex = (getCursor() != -1 ? getCursor() : getSelectedIndex()); if (selectedIndex < 0) selectedIndex = 0; int count = getElementCount(); if (count == 0) return s.length() > 0 ? -1 : 0; // find from cursor int found = text.lastIndexOf(str, selectedIndex); // if not found, start over from the end if (found == -1) { found = text.lastIndexOf(str, text.length() - 1); if (found <= selectedIndex) { found = -1; } } return found; } else { return super.findFromCursor(s); } } @Override public int findNext(String s) { if (_component instanceof JTextComponent) { String text = getDocumentText(); if (!isCaseSensitive()) { text = text.toLowerCase(); } String str = isCaseSensitive() ? s : s.toLowerCase(); int selectedIndex = (getCursor() != -1 ? getCursor() : getSelectedIndex()); if (selectedIndex < 0) selectedIndex = 0; int count = getElementCount(); if (count == 0) return s.length() > 0 ? -1 : 0; // find from cursor int found = text.indexOf(str, selectedIndex + 1); // if not found, start over from the beginning if (found == -1 && isRepeats()) { found = text.indexOf(str, 0); if (found >= selectedIndex) { found = -1; } } return found; } else { return super.findNext(s); } } @Override public int findPrevious(String s) { if (_component instanceof JTextComponent) { String text = getDocumentText(); if (!isCaseSensitive()) { text = text.toLowerCase(); } String str = isCaseSensitive() ? s : s.toLowerCase(); int selectedIndex = (getCursor() != -1 ? getCursor() : getSelectedIndex()); if (selectedIndex < 0) selectedIndex = 0; int count = getElementCount(); if (count == 0) return s.length() > 0 ? -1 : 0; // find from cursor int found = text.lastIndexOf(str, selectedIndex - 1); // if not found, start over from the beginning if (found == -1 && isRepeats()) { found = text.lastIndexOf(str, count - 1); if (found <= selectedIndex) { found = -1; } } return found; } else { return super.findPrevious(s); } } private class HighlighCache extends HashMap { public void addHighlight(Object obj) { put(obj, null); } public void removeHighlight(Object obj) { remove(obj); } public Iterator getAllHighlights() { return keySet().iterator(); } public void removeAllHighlights() { clear(); } } @Override public void hidePopup() { super.hidePopup(); _selectedIndex = -1; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy