2 * Jalview - A Sequence Alignment Editor and Viewer ($$Version-Rel$$)
3 * Copyright (C) $$Year-Rel$$ The Jalview Authors
5 * This file is part of Jalview.
7 * Jalview is free software: you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation, either version 3
10 * of the License, or (at your option) any later version.
12 * Jalview is distributed in the hope that it will be useful, but
13 * WITHOUT ANY WARRANTY; without even the implied warranty
14 * of MERCHANTABILITY or FITNESS FOR A PARTICULAR
15 * PURPOSE. See the GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License
18 * along with Jalview. If not, see <http://www.gnu.org/licenses/>.
19 * The Jalview Authors are detailed in the 'AUTHORS' file.
21 //////////////////////////////////////////////////////////////////
24 import jalview.bin.Cache;
25 import jalview.gui.JvOptionPane;
26 import jalview.util.MessageManager;
27 import jalview.util.Platform;
29 import java.awt.Component;
30 import java.awt.Dimension;
31 import java.awt.EventQueue;
32 import java.awt.HeadlessException;
33 import java.awt.event.ActionEvent;
34 import java.awt.event.ActionListener;
35 import java.awt.event.MouseAdapter;
36 import java.awt.event.MouseEvent;
38 import java.util.ArrayList;
39 import java.util.List;
40 import java.util.StringTokenizer;
41 import java.util.Vector;
43 import javax.swing.BoxLayout;
44 import javax.swing.DefaultListCellRenderer;
45 import javax.swing.JCheckBox;
46 import javax.swing.JFileChooser;
47 import javax.swing.JList;
48 import javax.swing.JPanel;
49 import javax.swing.JScrollPane;
50 import javax.swing.SpringLayout;
51 import javax.swing.filechooser.FileFilter;
52 import javax.swing.plaf.basic.BasicFileChooserUI;
55 * Enhanced file chooser dialog box.
57 * NOTE: bug on Windows systems when filechooser opened on directory to view
58 * files with colons in title.
63 public class JalviewFileChooser extends JFileChooser
66 * backupfilesCheckBox = "Include backup files" checkbox includeBackupfiles =
67 * flag set by checkbox
69 private JCheckBox backupfilesCheckBox = null;
71 protected boolean includeBackupFiles = false;
74 * Factory method to return a file chooser that offers readable alignment file
81 public static JalviewFileChooser forRead(String directory,
84 List<String> extensions = new ArrayList<>();
85 List<String> descs = new ArrayList<>();
86 for (FileFormatI format : FileFormats.getInstance().getFormats())
88 if (format.isReadable())
90 extensions.add(format.getExtensions());
91 descs.add(format.getName());
94 return new JalviewFileChooser(directory,
95 extensions.toArray(new String[extensions.size()]),
96 descs.toArray(new String[descs.size()]), selected, true);
100 * Factory method to return a file chooser that offers writable alignment file
107 public static JalviewFileChooser forWrite(String directory,
110 // TODO in Java 8, forRead and forWrite can be a single method
111 // with a lambda expression parameter for isReadable/isWritable
112 List<String> extensions = new ArrayList<>();
113 List<String> descs = new ArrayList<>();
114 for (FileFormatI format : FileFormats.getInstance().getFormats())
116 if (format.isWritable())
118 extensions.add(format.getExtensions());
119 descs.add(format.getName());
122 return new JalviewFileChooser(directory,
123 extensions.toArray(new String[extensions.size()]),
124 descs.toArray(new String[descs.size()]), selected, false);
127 public JalviewFileChooser(String dir)
129 super(safePath(dir));
130 setAccessory(new RecentlyOpened());
133 public JalviewFileChooser(String dir, String[] suffix, String[] desc,
136 this(dir, suffix, desc, selected, true);
140 * Constructor for a single choice of file extension and description
145 public JalviewFileChooser(String extension, String desc)
147 this(Cache.getProperty("LAST_DIRECTORY"), new String[] { extension },
149 { desc }, desc, true);
152 JalviewFileChooser(String dir, String[] extensions, String[] descs,
153 String selected, boolean allFiles)
155 this(dir, extensions, descs, selected, allFiles, false);
158 public JalviewFileChooser(String dir, String[] extensions, String[] descs,
159 String selected, boolean allFiles, boolean allowBackupFiles)
161 super(safePath(dir));
162 if (extensions.length == descs.length)
164 List<String[]> formats = new ArrayList<>();
165 for (int i = 0; i < extensions.length; i++)
167 formats.add(new String[] { extensions[i], descs[i] });
169 init(formats, selected, allFiles, allowBackupFiles);
173 System.err.println("JalviewFileChooser arguments mismatch: "
174 + extensions + ", " + descs);
178 private static File safePath(String dir)
185 File f = new File(dir);
186 if (f.getName().indexOf(':') > -1)
196 * a list of {extensions, description} for each file format
199 * if true, 'any format' option is included
201 void init(List<String[]> formats, String selected, boolean allFiles)
203 init(formats, selected, allFiles, false);
206 void init(List<String[]> formats, String selected, boolean allFiles,
207 boolean allowBackupFiles)
210 JalviewFileFilter chosen = null;
212 // SelectAllFilter needs to be set first before adding further
213 // file filters to fix bug on Mac OSX
214 setAcceptAllFileFilterUsed(allFiles);
216 for (String[] format : formats)
218 JalviewFileFilter jvf = new JalviewFileFilter(format[0], format[1]);
219 if (allowBackupFiles)
221 jvf.setParentJFC(this);
223 addChoosableFileFilter(jvf);
224 if ((selected != null) && selected.equalsIgnoreCase(format[1]))
232 setFileFilter(chosen);
235 if (allowBackupFiles)
237 JPanel multi = new JPanel();
238 multi.setLayout(new BoxLayout(multi, BoxLayout.PAGE_AXIS));
239 if (backupfilesCheckBox == null)
242 includeBackupFiles = Boolean.parseBoolean(
243 Cache.getProperty(BackupFiles.NS + "_FC_INCLUDE"));
244 } catch (Exception e)
246 includeBackupFiles = false;
248 backupfilesCheckBox = new JCheckBox(
249 MessageManager.getString("label.include_backup_files"),
251 backupfilesCheckBox.setAlignmentX(Component.CENTER_ALIGNMENT);
252 JalviewFileChooser jfc = this;
253 backupfilesCheckBox.addActionListener(new ActionListener()
256 public void actionPerformed(ActionEvent e)
258 includeBackupFiles = backupfilesCheckBox.isSelected();
259 Cache.setProperty(BackupFiles.NS + "_FC_INCLUDE",
260 String.valueOf(includeBackupFiles));
262 if (Platform.isAMac())
264 // This is a kludge. Cannot find out how to get the file list to
265 // refresh on its own! This works the best out of a number of
266 // different attempts!
267 FileFilter ff = jfc.getFileFilter();
268 jfc.setFileFilter(jfc.getAcceptAllFileFilter());
269 jfc.setFileFilter(ff);
272 jfc.setFileHidingEnabled(!jfc.isFileHidingEnabled());
273 jfc.setFileHidingEnabled(!jfc.isFileHidingEnabled());
278 .paintImmediately(jfc.getRootPane().getBounds());
279 jfc.paintImmediately(jfc.getBounds());
283 Object o = jfc.getClientProperty(
284 CHOOSABLE_FILE_FILTER_CHANGED_PROPERTY);
285 jfc.firePropertyChange(CHOOSABLE_FILE_FILTER_CHANGED_PROPERTY,
290 File f = jfc.getSelectedFile();
291 jfc.setSelectedFile(null);
294 jfc.setSelectedFile(f);
299 Graphics g = jfc.getGraphics();
305 jfc.setFileHidingEnabled(!jfc.isFileHidingEnabled());
306 jfc.setFileHidingEnabled(!jfc.isFileHidingEnabled());
310 Component[] c = jfc.getComponents();
311 for (int i = 0; i < c.length; i++)
314 System.out.println("INVALIDATING " + c[i].getName());
319 jfc.rescanCurrentDirectory();
325 multi.add(new RecentlyOpened());
326 multi.add(backupfilesCheckBox);
331 // set includeBackupFiles=false to avoid other file choosers from picking
332 // up backup files (Just In Case)
333 includeBackupFiles = false;
334 setAccessory(new RecentlyOpened());
339 public void setFileFilter(javax.swing.filechooser.FileFilter filter)
341 super.setFileFilter(filter);
345 if (getUI() instanceof BasicFileChooserUI)
347 final BasicFileChooserUI fcui = (BasicFileChooserUI) getUI();
348 final String name = fcui.getFileName().trim();
350 if ((name == null) || (name.length() == 0))
355 EventQueue.invokeLater(new Thread()
360 String currentName = fcui.getFileName();
361 if ((currentName == null) || (currentName.length() == 0))
363 fcui.setFileName(name);
368 } catch (Exception ex)
370 ex.printStackTrace();
371 // Some platforms do not have BasicFileChooserUI
376 * Returns the selected file format, or null if none selected
380 public FileFormatI getSelectedFormat()
382 if (getFileFilter() == null)
388 * logic here depends on option description being formatted as
389 * formatName (extension, extension...)
390 * or the 'no option selected' value
392 * @see JalviewFileFilter.getDescription
394 String format = getFileFilter().getDescription();
395 int parenPos = format.indexOf("(");
398 format = format.substring(0, parenPos).trim();
401 return FileFormats.getInstance().forName(format);
402 } catch (IllegalArgumentException e)
404 System.err.println("Unexpected format: " + format);
410 File ourselectedFile = null;
413 public File getSelectedFile()
415 File selfile = super.getSelectedFile();
416 if (selfile == null && ourselectedFile != null)
418 return ourselectedFile;
424 public int showSaveDialog(Component parent) throws HeadlessException
426 this.setAccessory(null);
428 setDialogType(SAVE_DIALOG);
430 this.setSelectedFile(null);
431 int ret = showDialog(parent, MessageManager.getString("action.save"));
432 ourselectedFile = getSelectedFile();
434 if (getSelectedFile() == null)
436 // Workaround for Java 9,10 on OSX - no selected file, but there is a
440 String filename = ((BasicFileChooserUI) getUI()).getFileName();
441 if (filename != null && filename.length() > 0)
443 ourselectedFile = new File(getCurrentDirectory(), filename);
445 } catch (Throwable x)
448 "Unexpected exception when trying to get filename.");
452 if (ourselectedFile == null)
454 return JalviewFileChooser.CANCEL_OPTION;
456 if (getFileFilter() instanceof JalviewFileFilter)
458 JalviewFileFilter jvf = (JalviewFileFilter) getFileFilter();
460 if (!jvf.accept(ourselectedFile))
462 String withExtension = getSelectedFile().getName() + "."
463 + jvf.getAcceptableExtension();
464 ourselectedFile = (new File(getCurrentDirectory(), withExtension));
465 setSelectedFile(ourselectedFile);
469 // TODO: ENSURE THAT FILES SAVED WITH A ':' IN THE NAME ARE REFUSED AND THE
470 // USER PROMPTED FOR A NEW FILENAME.
471 // DO NOT need to confirm file overwrite if using backup files (the files
472 // aren't being overwritten!)
473 if ((ret == JalviewFileChooser.APPROVE_OPTION)
474 && ourselectedFile.exists() && (!BackupFiles.getEnabled()))
476 int confirm = JvOptionPane.showConfirmDialog(parent,
477 MessageManager.getString("label.overwrite_existing_file"),
478 MessageManager.getString("label.file_already_exists"),
479 JvOptionPane.YES_NO_OPTION);
481 if (confirm != JvOptionPane.YES_OPTION)
483 ret = JalviewFileChooser.CANCEL_OPTION;
491 void recentListSelectionChanged(Object selection)
493 setSelectedFile(null);
494 if (selection != null)
496 File file = new File((String) selection);
497 if (getFileFilter() instanceof JalviewFileFilter)
499 JalviewFileFilter jvf = (JalviewFileFilter) this.getFileFilter();
501 if (!jvf.accept(file))
503 setFileFilter(getChoosableFileFilters()[0]);
507 setSelectedFile(file);
511 class RecentlyOpened extends JPanel
515 public RecentlyOpened()
518 String historyItems = jalview.bin.Cache.getProperty("RECENT_FILE");
520 Vector recent = new Vector();
522 if (historyItems != null)
524 st = new StringTokenizer(historyItems, "\t");
526 while (st.hasMoreTokens())
528 recent.addElement(st.nextElement());
532 list = new JList(recent);
534 DefaultListCellRenderer dlcr = new DefaultListCellRenderer();
535 dlcr.setHorizontalAlignment(DefaultListCellRenderer.RIGHT);
536 list.setCellRenderer(dlcr);
538 list.addMouseListener(new MouseAdapter()
541 public void mousePressed(MouseEvent evt)
543 recentListSelectionChanged(list.getSelectedValue());
547 this.setBorder(new javax.swing.border.TitledBorder(
548 MessageManager.getString("label.recently_opened")));
550 final JScrollPane scroller = new JScrollPane(list);
552 SpringLayout layout = new SpringLayout();
553 layout.putConstraint(SpringLayout.WEST, scroller, 5,
554 SpringLayout.WEST, this);
555 layout.putConstraint(SpringLayout.NORTH, scroller, 5,
556 SpringLayout.NORTH, this);
558 if (new Platform().isAMac())
560 scroller.setPreferredSize(new Dimension(500, 100));
564 scroller.setPreferredSize(new Dimension(130, 200));
569 javax.swing.SwingUtilities.invokeLater(new Runnable()
574 scroller.getHorizontalScrollBar()
575 .setValue(scroller.getHorizontalScrollBar().getMaximum());