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

de.schlichtherle.truezip.swing.AbstractComboBoxBrowser Maven / Gradle / Ivy

/*
 * Copyright (C) 2005-2013 Schlichtherle IT Services.
 * All rights reserved. Use is subject to license terms.
 */
package de.schlichtherle.truezip.swing;

import java.awt.Component;
import java.awt.event.ActionListener;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.io.Serializable;
import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
import javax.annotation.concurrent.NotThreadSafe;
import javax.swing.ComboBoxEditor;
import javax.swing.ComboBoxModel;
import javax.swing.JComboBox;
import javax.swing.MutableComboBoxModel;
import javax.swing.event.DocumentEvent;
import javax.swing.event.DocumentListener;
import javax.swing.text.BadLocationException;
import javax.swing.text.Caret;
import javax.swing.text.Document;
import javax.swing.text.JTextComponent;

/**
 * An observer for a {@link JComboBox} which provides auto completion for the
 * editable text in the drop down list in order to provide quick browsing
 * capabilities for the user. Subclasses need to implement the {@link #update}
 * method in order to update the combo box model with the actual auto
 * completion data.
 * 

* This class is designed to be minimal intrusive: It's designed to work with * any {@code JComboBox} and doesn't require a special {@link ComboBoxModel}, * although its specific behaviour will only show if the {@code JComboBox} is * {@code editable} and uses a {@link MutableComboBoxModel} (which, besides the * {@code editable} property being set to {@code true}, is the default setup * for a vanilla {@code JComboBox}). * * @param the type of the elements of this combo box browser. * @author Christian Schlichtherle */ @NotThreadSafe public abstract class AbstractComboBoxBrowser implements Serializable { private static final long serialVersionUID = 1065103960246722893L; private final Listener listener = new Listener(); private @CheckForNull JComboBox comboBox; /** * Used to inhibit mutual recursive event firing. */ private transient boolean recursion; /** * Creates a new combo box auto completion browser. * {@link #setComboBox} must be called in order to use this object. */ protected AbstractComboBoxBrowser() { } /** * Creates a new combo box auto completion browser. Note that this * constructor does not call {@link #update} and hence the drop * down list of the combo box is left unchanged. * * @param comboBox The combo box to enable browsing for auto completions. * May be {@code null}. */ protected AbstractComboBoxBrowser(final @CheckForNull JComboBox comboBox) { changeComboBox(null, comboBox, false); } /** * Returns the combo box which this object is auto completing. * The default is {@code null}. * * @return The combo box which this object is auto completing. */ public @Nullable JComboBox getComboBox() { return comboBox; } /** * Sets the combo box which this object is auto completing and updates the * drop down list with the auto completion for the currently selected item. * * @param comboBox The combo box to enable browsing for auto completions. * May be {@code null}. */ public void setComboBox(final @CheckForNull JComboBox comboBox) { changeComboBox(getComboBox(), comboBox, true); } private void changeComboBox( final @CheckForNull JComboBox oldCB, final @CheckForNull JComboBox newCB, final boolean update) { if (newCB == oldCB) return; ComboBoxEditor oldCBE = null; if (null != oldCB) { oldCB.removePropertyChangeListener("editor", listener); oldCBE = oldCB.getEditor(); oldCB.setEditor(((AbstractComboBoxBrowser.DecoratingComboBoxEditor) oldCBE).getEditor()); } this.comboBox = newCB; ComboBoxEditor newCBE = null; if (null != newCB) { newCB.updateUI(); // ensure comboBoxEditor is initialized newCBE = new DecoratingComboBoxEditor(newCB.getEditor()); newCB.setEditor(newCBE); newCB.addPropertyChangeListener("editor", listener); } changeEditor(oldCBE, newCBE, update); } private void changeEditor( final @CheckForNull ComboBoxEditor oldCBE, final @CheckForNull ComboBoxEditor newCBE, final boolean update) { if (newCBE == oldCBE) return; JTextComponent oldText = null; if (null != oldCBE) { final Component component = oldCBE.getEditorComponent(); if (component instanceof JTextComponent) oldText = (JTextComponent) component; } JTextComponent newText = null; if (null != newCBE) { final Component component = newCBE.getEditorComponent(); if (component instanceof JTextComponent) newText = (JTextComponent) component; } changeText(oldText, newText, update); } private void changeText( final @CheckForNull JTextComponent oldTC, final @CheckForNull JTextComponent newTC, final boolean update) { if (newTC == oldTC) return; Document oldDocument = null; if (null != oldTC) { oldTC.removePropertyChangeListener("document", listener); oldDocument = oldTC.getDocument(); } Document newDocument = null; if (null != newTC) { newDocument = newTC.getDocument(); newTC.addPropertyChangeListener("document", listener); } changeDocument(oldDocument, newDocument, update); } private void changeDocument( final @CheckForNull Document oldDoc, final @CheckForNull Document newDoc, final boolean update) { if (newDoc == oldDoc) return; if (null != oldDoc) oldDoc.removeDocumentListener(listener); if (null != newDoc) { if (update) { String txt; try { txt = newDoc.getText(0, newDoc.getLength()); } catch (BadLocationException e) { txt = null; } update(txt); } newDoc.addDocumentListener(listener); } } private void documentUpdated() { if (lock()) return; try { final JComboBox cb = getComboBox(); final ComboBoxEditor cbe = cb.getEditor(); final JTextComponent tc = (JTextComponent) cbe.getEditorComponent(); assert cb.isShowing() || !tc.isFocusOwner(); if (!tc.isFocusOwner() /* || !cb.isShowing() */) return; //cb.setPopupVisible(update(tc.getText())); // doesn't work: adjusts popup size! cb.setPopupVisible(false); if (update(tc.getText())) cb.setPopupVisible(true); } finally { unlock(); } } private void updateEditor(final ComboBoxEditor cbe, final @CheckForNull Object item) { if (lock()) return; try { cbe.setItem(item); if (!(item instanceof String)) return; final JComboBox cb = getComboBox(); final JTextComponent tc = (JTextComponent) cbe.getEditorComponent(); assert cb.isShowing() || !tc.isFocusOwner(); if (!tc.isFocusOwner() /* || !cb.isShowing() */) return; // Compensate for an issue with some look and feels // which select the entire tc if an item is changed. // This is inconvenient for auto completion because the // next typed character would replace the entire tc... final Caret caret = tc.getCaret(); caret.setDot(((String) item).length()); } finally { unlock(); } } /** * Subclasses are expected to update the auto completion elements in the * model of this combo box based on the specified {@code initials}. * They should not do any other work within this method. * In particular, they should not update the visual appearance of this * component. *

* {@link #getComboBox} is guaranteed to return non-{@code null} if this * method is called from this abstract base class. * * @param initials The text to auto complete. May be {@code null}. * @return Whether or not the combo box should pop up to show the updated * contents of its model. */ protected abstract boolean update(@CheckForNull String initials); /** * Locks out mutual recursive event notification. * Warning: This method works in a synchronized or single threaded * environment only! * * @return Whether or not updating the combo box model was already locked. */ private boolean lock() { if (recursion) return true; recursion = true; return false; } /** * Unlocks mutual recursive event notification. * Warning: This method works in a synchronized or single threaded * environment only! */ private void unlock() { recursion = false; } private class Listener implements DocumentListener, PropertyChangeListener { @Override public void insertUpdate(DocumentEvent e) { documentUpdated(); } @Override public void removeUpdate(DocumentEvent e) { documentUpdated(); } @Override public void changedUpdate(DocumentEvent e) { documentUpdated(); } @Override public void propertyChange(final PropertyChangeEvent e) { final String property = e.getPropertyName(); if ("editor".equals(property)) changeEditor( (ComboBoxEditor) e.getOldValue(), (ComboBoxEditor) e.getNewValue(), true); else if ("document".equals(property)) changeDocument( (Document) e.getOldValue(), (Document) e.getNewValue(), true); else throw new AssertionError( "Received change event for unknown property: " + property); } } /** * This proxy controls access to the real {@code ComboBoxEditor} installed * by the client application or the pluggable look and feel. * It is used to inhibit mutual recursion caused by modifications to the * list model in the {@code JComboBox}. */ private final class DecoratingComboBoxEditor implements ComboBoxEditor { private final ComboBoxEditor delegate; DecoratingComboBoxEditor(ComboBoxEditor comboBoxEditor) { assert null != comboBoxEditor; this.delegate = comboBoxEditor; } /** Returns the decorated combo box editor. */ ComboBoxEditor getEditor() { return delegate; } @Override public Component getEditorComponent() { return delegate.getEditorComponent(); } @Override public void setItem(final @CheckForNull Object item) { updateEditor(delegate, item); } @Override public @CheckForNull Object getItem() { return delegate.getItem(); } @Override public void selectAll() { delegate.selectAll(); } @Override public void addActionListener(ActionListener actionListener) { delegate.addActionListener(actionListener); } @Override public void removeActionListener(ActionListener actionListener) { delegate.removeActionListener(actionListener); } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy