001/** 002 * BioJava development code 003 * 004 * This code may be freely distributed and modified under the 005 * terms of the GNU Lesser General Public Licence. This should 006 * be distributed with the code. If you do not have a copy, 007 * see: 008 * 009 * http://www.gnu.org/copyleft/lesser.html 010 * 011 * Copyright for this code is held jointly by the individual 012 * authors. These should be listed in @author doc comments. 013 * 014 * For more information on the BioJava project and its aims, 015 * or to join the biojava-l mailing list, visit the home page 016 * at: 017 * 018 * http://www.biojava.org/ 019 * 020 * Created on Sep 14, 2011 021 * Created by Andreas Prlic 022 * 023 * @since 3.0.2 024 */ 025package org.biojava.nbio.structure.align.gui.autosuggest; 026 027import javax.swing.*; 028import java.awt.*; 029import java.awt.event.*; 030import java.util.Vector; 031import java.util.concurrent.atomic.AtomicBoolean; 032 033 034 035 036/** A JTextField that can make suggestions for auto-complete. 037 * 038 * @author Andreas Prlic 039 * 040 */ 041public class JAutoSuggest extends JTextField{ 042 043 /** 044 * 045 */ 046 private static final long serialVersionUID = 8591734727984365156L; 047 048 private static final String DEFAULT_TEXT= "Please enter text ..."; 049 050 String defaultText; 051 private JDialog dialog; 052 private Point location; 053 private JList list; 054 055 private Vector<String> suggestions; 056 057 /** last word that was entered by user */ 058 private String lastWord ; 059 060 AutoSuggestProvider autoSuggestProvider; 061 062 Font regular; 063 Font busy; 064 065 SuggestionFetcher matcher; 066 067 public JAutoSuggest(){ 068 super(); 069 070 init(); 071 } 072 073 public JAutoSuggest(int size){ 074 super(size); 075 init(); 076 } 077 078 public JAutoSuggest(Frame owner){ 079 owner.addComponentListener(new ComponentListener() { 080 @Override 081 public void componentShown(ComponentEvent e) { 082 updateLocation(); 083 } 084 085 @Override 086 public void componentResized(ComponentEvent e) { 087 updateLocation(); 088 } 089 090 @Override 091 public void componentMoved(ComponentEvent e) { 092 updateLocation(); 093 } 094 095 @Override 096 public void componentHidden(ComponentEvent e) { 097 updateLocation(); 098 } 099 }); 100 owner.addWindowListener(new WindowListener() { 101 @Override 102 public void windowOpened(WindowEvent e) { 103 } 104 105 @Override 106 public void windowIconified(WindowEvent e) { 107 dialog.setVisible(false); 108 } 109 110 @Override 111 public void windowDeiconified(WindowEvent e) { 112 } 113 114 @Override 115 public void windowDeactivated(WindowEvent e) { 116 } 117 118 @Override 119 public void windowClosing(WindowEvent e) { 120 dialog.dispose(); 121 } 122 123 @Override 124 public void windowClosed(WindowEvent e) { 125 dialog.dispose(); 126 } 127 128 @Override 129 public void windowActivated(WindowEvent e) { 130 } 131 }); 132 addFocusListener(new FocusListener() { 133 @Override 134 public void focusLost(FocusEvent e) { 135 System.out.println("Lost Focus"); 136 dialog.setVisible(false); 137 138 if ("".equals(getText().trim()) && e.getOppositeComponent() != null && e.getOppositeComponent().getName() != null) { 139 if (!"suggestFieldDropdownButton".equals(e.getOppositeComponent().getName())) { 140 setText(defaultText); 141 } 142 } else if ("".equals(getText().trim())) { 143 setText(defaultText); 144 } 145 } 146 147 @Override 148 public void focusGained(FocusEvent e) { 149 System.out.println("Lost Gained"); 150 if (getText().trim().equals(defaultText)) { 151 setText(""); 152 } 153 154 showSuggest(); 155 } 156 }); 157 158 159 160 161 init(); 162 163 // set dialog owner... 164 //dialog.setD 165 166 } 167 168 169 private void initSuggestionList() { 170 list = new JList(); 171 list.addMouseListener(new MouseListener() { 172 private int selected; 173 174 @Override 175 public void mousePressed(MouseEvent e) { 176 } 177 178 @Override 179 public void mouseReleased(MouseEvent e) { 180 if (selected == list.getSelectedIndex()) { 181 // provide double-click for selecting a suggestion 182 setText((String) list.getSelectedValue()); 183 184 dialog.setVisible(false); 185 } 186 selected = list.getSelectedIndex(); 187 } 188 189 @Override 190 public void mouseExited(MouseEvent e) { 191 } 192 193 @Override 194 public void mouseEntered(MouseEvent e) { 195 } 196 197 @Override 198 public void mouseClicked(MouseEvent e) { 199 } 200 }); 201 dialog.add(new JScrollPane(list, JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED, 202 JScrollPane.HORIZONTAL_SCROLLBAR_NEVER)); 203 dialog.pack(); 204 addKeyListener(new KeyListener() { 205 @Override 206 public void keyTyped(KeyEvent e) { 207 } 208 209 @Override 210 public void keyPressed(KeyEvent e) { 211 updateLocation(); 212 } 213 214 @Override 215 public void keyReleased(KeyEvent e) { 216 if (e.getKeyCode() == KeyEvent.VK_ESCAPE) { 217 dialog.setVisible(false); 218 return; 219 } else if (e.getKeyCode() == KeyEvent.VK_DOWN) { 220 if (dialog.isVisible()) { 221 list.setSelectedIndex(list.getSelectedIndex() + 1); 222 list.ensureIndexIsVisible(list.getSelectedIndex() + 1); 223 return; 224 } else { 225 showSuggest(); 226 } 227 } else if (e.getKeyCode() == KeyEvent.VK_UP) { 228 list.setSelectedIndex(list.getSelectedIndex() - 1); 229 list.ensureIndexIsVisible(list.getSelectedIndex() - 1); 230 return; 231 } else if (e.getKeyCode() == KeyEvent.VK_ENTER 232 && list.getSelectedIndex() != -1 && suggestions.size() > 0) { 233 setText((String) list.getSelectedValue()); 234 235 236 dialog.setVisible(false); 237 return; 238 } 239 showSuggest(); 240 } 241 }); 242 243 } 244 245 private void init(){ 246 autoSuggestProvider = new DefaultAutoSuggestProvider(); 247 lastWord = ""; 248 regular = getFont(); 249 busy = new Font(getFont().getName(), Font.ITALIC, getFont().getSize()); 250 suggestions = new Vector<>(); 251 defaultText = DEFAULT_TEXT; 252 253 254 dialog = new JDialog(); 255 dialog.setUndecorated(true); 256 dialog.setFocusableWindowState(false); 257 dialog.setFocusable(false); 258 259 initSuggestionList(); 260 } 261 262 public String getDefaultText() { 263 return defaultText; 264 } 265 266 public void setDefaultText(String defaultText) { 267 this.defaultText = defaultText; 268 } 269 270 public AutoSuggestProvider getAutoSuggestProvider() { 271 return autoSuggestProvider; 272 } 273 274 public void setAutoSuggestProvider(AutoSuggestProvider autoSuggestProvider) { 275 this.autoSuggestProvider = autoSuggestProvider; 276 } 277 278 /** 279 * Force the suggestions to be displayed (Useful for buttons 280 * e.g. for using JSuggestionField like a ComboBox) 281 */ 282 public void showSuggest() { 283 284 assert(getText() != null); 285 lastWord = getText().trim(); 286 //autoSuggestProvider.getSuggestion(lastWord); 287 288 289 290 if (!getText().toLowerCase().contains(lastWord.toLowerCase())) { 291 suggestions.clear(); 292 } 293 294 if (matcher != null) { 295 matcher.setStop(); 296 } 297 matcher = new SuggestionFetcher(); 298 299 //SwingUtilities.invokeLater(matcher); 300 matcher.execute(); 301 lastWord = getText().trim(); 302 updateLocation(); 303 } 304 305 /** 306 * Force the suggestions to be hidden (Useful for buttons, e.g. to use 307 * JSuggestionField like a ComboBox) 308 */ 309 public void hideSuggest() { 310 dialog.setVisible(false); 311 } 312 313 /** 314 * @return boolean Visibility of the suggestion window 315 */ 316 public boolean isSuggestVisible() { 317 return dialog.isVisible(); 318 } 319 320 321 322 /** 323 * Place the suggestion window under the JTextField. 324 */ 325 private void updateLocation() { 326 try { 327 location = getLocationOnScreen(); 328 location.y += getHeight(); 329 dialog.setLocation(location); 330 } catch (IllegalComponentStateException e) { 331 return; // might happen on window creation 332 } 333 } 334 335 336 /** fetch suggestions from SuggestionProvider 337 * 338 * 339 * 340 */ 341 private class SuggestionFetcher extends SwingWorker<String, Object> { 342 /** flag used to stop the thread */ 343 private AtomicBoolean stop = new AtomicBoolean(false); 344 345 String previousWord; 346 /** 347 * Standard run method used in threads 348 * responsible for the actual search 349 */ 350 @Override 351 public String doInBackground() { 352 try { 353 setFont(busy); 354 String userInput = getText().trim(); 355 if ( userInput == null || "".equals(userInput)) 356 return ""; 357 358 if ( previousWord != null){ 359 if ( userInput.equals(previousWord)) 360 return ""; 361 } 362 previousWord = userInput; 363 364 suggestions = autoSuggestProvider.getSuggestion(userInput); 365 366 setFont(regular); 367 368 if (suggestions.size() > 0) { 369 list.setListData(suggestions); 370 list.setSelectedIndex(0); 371 list.ensureIndexIsVisible(0); 372 dialog.setVisible(true); 373 } else { 374 dialog.setVisible(false); 375 } 376 } catch (Exception e) { 377 //e.printStackTrace(); 378 // ignore... 379 380 } 381 return "Done."; 382 } 383 384 public void setStop(){ 385 stop.set(true); 386 } 387 388 389 } 390}