我们在做GUI编程的时候经常需要用到JFileChooser组件构造一个文件选取对话框来为用户提供打开文件、保存文件等操作。
通常的做法是调用JFileChooser.showXXX()方法显示文件选取对话框并且选择一个文件后,点击Approve 按钮(默认情况下标有 "Open" 或 "Save"),当对话框关闭后使用JFileChooser.getSelectedFile()方法得到选取的文件(或使用JFileChooser.getSelectedFiles()取得选取的文件数组),然后再对被选取的文件有效性进行验证(例如,文件的文件名是否合法、选取的路径下是否已有同名文件存在等等),如果验证不通过,需要再次打开文件选择对话框进行选取。
显然,验证没有通过的情况下,文件选取对话框被反复的打开和关闭,影响用户体验。
我现在介绍一个方法,在点击了文件选取对话框上的Approve 按钮后,文件选取对话框关闭前对选取的文件进行验证,如果验证没有通过,那么对话框不关闭,直接进行下一次选择。
查看JFileChooser的API,可以发现这样一个方法 public void approveSelection() ,这个方法会在用户单击 Approve 按钮时由 UI 调用此方法。导致使用等于 APPROVE_SELECTION
的命令字符串激发一个操作事件。
那么,我们现在可以选择继承JFileChooser再覆写这个方法,将对选中文件的有效性验证写入这个方法中,只有当验证通过时才调用超类的approveSelection() 完成文件选取,否则直接返回,继续选择新的文件。
下面是我写的一个demo以供参考:
import java.io.File;
import javax.swing.JOptionPane;
/**
* 在 JFileChooser 中进行文件验证的小技巧
* @author Chen Wei
* @email chenwei.mobi@gmail.com
*/
public class JFileChooserDemo extends javax.swing.JFileChooser{
public void approveSelection(){
File file = getSelectedFile();
// 验证文件名是否合法
if (!validateFileName(file.getName())) {
JOptionPane.showMessageDialog(getParent(), "文件名不能包含下列任何字符之一:\n \\ / : * ? \" < > |");
return;
}else{
super.approveSelection();
}
}
/**
* 验证输入字符串参数是否为有效文件名。
* @param name 待验证的文件名字符串。
* @return 通过验证,文件名无效返回 false,有效返回 true。
*/
public static boolean validateFileName(String name) {
if (name.indexOf('\\') != -1 || name.indexOf('/') != -1 ||
name.indexOf(':') != -1 || name.indexOf('*') != -1 ||
name.indexOf('?') != -1 || name.indexOf('"') != -1 ||
name.indexOf('<') != -1 || name.indexOf('>') != -1 ||
name.indexOf('|') != -1) {
return false;
} else {
return true;
}
}
public static void main(String[] args){
JFileChooserDemo chooser = new JFileChooserDemo();
chooser.showOpenDialog(null);
}
}
程序运行截图: