在这篇文章中我接着介绍另外几种新的GUI功能
这些功能是
·带有排序和过滤功能的JTable
·增强的JTabbedPane组件
·增强的打印功能
·增强的拖放功能
带有排序和过滤功能的JTable
在Java SE 中除了javaawt被更新外javaxswing同时也有了很大的改进在C/S程序中我们会经常使用到表如我们可以在查询数据库后将查询结果显示在表格中在Java中显示表格使用的是JTable类在以前的版本中JTable只能简单地显示数据并没有什么附加的处理功能而在Java SE 中的JTable增加了排序和过滤功能用户可以单击列头进行排序也可以根据某一列来过滤表中的数据
为了使JTable可以对数据进行必须将RowSorter类和JTable进行关联RowSorter是一个抽象类它负责将JTable中的数据映射成可排序的数据在真正使用时我们将直接使用RowSorter的子类TableRowSorter下面的代码显示了如何将TableRowSorter类和JTable相关联
TableModel model = new DefaultTableModel(rows columns);
JTable table = new JTable(model);
RowSorter sorter = new TableRowSorter(model);
tablesetRowSorter(sorter);
上面代码首先建立一个TableModel然后将这个TableModel的实例同时传递给了JTable和RowSorter下面是一个使用JTable排序的简单的例子
import javaxswing*;
import javaxswingtable*;
import javaawt*;
public class TestSortedTable
{
public static void main(String args[])
{
JFrame frame = new JFrame(JTable的排序测试);
framesetDefaultCloseOperation(JFrameEXIT_ON_CLOSE);
// 表格中显示的数据
Object rows[][] =
{
{ 王明 中国 }
{ 姚明 中国 }
{ 赵子龙 西蜀 }
{ 曹操 北魏 }
{ Bill Gates 美国 }
{ Mike 英国 } };
String columns[] =
{ 姓名 国籍 年龄 };
TableModel model = new DefaultTableModel(rows columns);
JTable table = new JTable(model);
RowSorter<TableModel> sorter = new TableRowSorter<TableModel>(model);
tablesetRowSorter(sorter);
JScrollPane pane = new JScrollPane(table);
frameadd(pane BorderLayoutCENTER);
framesetSize( );
framesetVisible(true);
}
}
图和图分别是按姓名进行升序和降序排列的显示结果
图 按姓名升序显示
图 按姓名降序显示
图显示的是按年龄进行降序排列但我们发现一个奇怪的问题就是年龄字段并不是按数值类型进行排序的而是按字符类型进行排序的
图 按年龄降序显示
出现这种情况是因为在默认情况下DefaultTableModal的列是Object类型而要想使JTable按数值进行排序必须要覆盖DefaultTableModal的getColumnClass方法
TableModel model = new DefaultTableModel(rows
columns)
{
public Class getColumnClass(int column)
{
Class returnValue;
if ((column >= ) && (column < getColumnCount()))
{
returnValue = getValueAt( column)getClass();
}
else
{
returnValue = Objectclass;
}
return returnValue;
}
};
图显示了按年龄进行排序的界面看看是不是按数值进行排序了
图 按数值类型进行排序
下面让我们来看看来何使用JTable进行过滤我们可以通过convertRowIndexToModel方法进行过滤下面的代码加在一个按钮中添加事件代码调用JTable的过滤功能
buttonaddActionListener(new ActionListener()
{
public void actionPerformed(ActionEvent e)
{
String text = filterTextgetText();
if (textlength() == )
{
sortersetRowFilter(null);
}
else
{
sortersetRowFilter(RowFilterregexFilter(text));
}
}
});
上面的代码并没有调用convertRowIndextoModel()方法如果调用它你就可以在表中进行相应的操作
在JTable中通过抽象类RowFilter类对行进行过滤和排序不同你可以不建立它们的子类而使用这个抽象类的个静态方法
·andFilter
·dateFilter(RowFilterComparisonType type Date date int indices)
·notFilter(RowFilter<MI> filter)
·numberFilter(RowFilterComparisonType type Number number int indices)
·orFilter
·regexFilter(String regex int indices)
其中andFilter()orFilter()以及notFilter()方法的功能是将当前的过滤条件和其它的过滤条件进行组合如在同时比较日期和数值时需要将日期过滤和数值过滤进行组合这些组合是非常简单的
RowFilter的类型比较允许你进行种关系的比较等于不等于大于或小于我们可以通过指定某一列进行过滤也可以对所有的列进行过滤这其中最为有趣的也许是正则表达式过滤(regular expression filter或简称为regex filter)使用这个过滤器可以对表中数据进行更高级的过滤下面是实现一个简单过滤器的代码
import javaxswing*;
import javaxswingtable*;
import javaawt*;
import javaawtevent*;
public class TestFilter
{
public static void main(String args[])
{
JFrame frame = new JFrame(JTable的过滤测试);
framesetDefaultCloseOperation(JFrameEXIT_ON_CLOSE);
Object rows[][] =
{
{ 王明 中国 }
{ 姚明 中国 }
{ 赵子龙 西蜀 }
{ 曹操 北魏 }
{ Bill Gates 美国 }
{ Mike 英国 } };
String columns[] =
{ 姓名 国籍 年龄 };
TableModel model = new DefaultTableModel(rows columns)
{
public Class getColumnClass(int column)
{
Class returnValue;
if ((column >= ) && (column < getColumnCount()))
{
returnValue = getValueAt( column)getClass();
}
else
{
returnValue = Objectclass;
}
return returnValue;
}
};
final JTable table = new JTable(model);
final TableRowSorter<TableModel> sorter = new TableRowSorter<TableModel>(model);
tablesetRowSorter(sorter);
JScrollPane pane = new JScrollPane(table);
frameadd(pane BorderLayoutCENTER);
JPanel panel = new JPanel(new BorderLayout());
JLabel label = new JLabel(过滤);
paneladd(label BorderLayoutWEST);
final JTextField filterText = new JTextField();
paneladd(filterText BorderLayoutCENTER);
frameadd(panel BorderLayoutNORTH);
JButton button = new JButton(过滤);
buttonaddActionListener(new ActionListener()
{
public void actionPerformed(ActionEvent e)
{
String text = filterTextgetText();
if (textlength() == )
{
sortersetRowFilter(null);
}
else
{
sortersetRowFilter(RowFilterregexFilter(text));
}
}
});
frameadd(button BorderLayoutSOUTH);
framesetSize( );
framesetVisible(true);
}
}
图是上面程序的运行界面
图
增强的JTabbedPane组件
JTabbedPane组件为我们提供了一种非常好的方法在窗体上显示很多的控件我们可以将不同类别的控件放到不同的Tab页上然后通过需要点击相应的Tab页在传统的Tab页上只能防止文本的图标而在Java SE 中使我们可以直接将控件放到Tab上我们可以通过setTabComponentAt方法将控件放到Tab上这个方法有两个参数一个是Tab的索引另一个是要放置的对象
JTabbedPane pane = new JTabbedPane();
panesetTabComponentAt( component);
在JTabbedPane控件中有个常用的方法setTabComponentAt(int index Component comp) getTabComponentAt(int index)和indexOfTabComponent(Component)最后一个方法将替换Tab上的控件下面的代码是一个关于JTabbedPane控件的演示
import javaxswing*;
import javaxswingtable*;
import javaawt*;
import javaawtevent*;
public class TestTabbedPane
{
static void addIt(JTabbedPane tabbedPane String text)
{
JLabel label = new JLabel(text);
JButton button = new JButton(text);
JPanel panel = new JPanel();
paneladd(label);
paneladd(button);
tabbedPaneaddTab(text panel);
if(textequals(tab))
tabbedPanesetTabComponentAt(tabbedPanegetTabCount()
new JTextField(插入了文本控件));
else
tabbedPanesetTabComponentAt(tabbedPanegetTabCount() button);
}
public static void main(String args[])
{
JFrame f = new JFrame(JTabbedPane演示);
fsetDefaultCloseOperation(JFrameEXIT_ON_CLOSE);
JTabbedPane tabbedPane = new JTabbedPane();
addIt(tabbedPane tab);
addIt(tabbedPane tab);
addIt(tabbedPane tab);
addIt(tabbedPane tab);
addIt(tabbedPane tab);
fadd(tabbedPane BorderLayoutCENTER);
fsetSize( );
fsetVisible(true);
}
}
图是显示界面其中在Tab上插入了一个文本控件在Tab至Tab上各插入了一个按钮控件
图 JTabbedPane演示 增强的打印功能
自从Java SE 开始Sun就对控件的打印功能进行了加强如JTextFieldJTextArea等在Java SE 中Sun为打印增加了分页功能我们只需要调用JtextField或JTextArea的print方法就可以调用打印对话框下面是一段测试代码
import javaxswing*;
import javaawt*;
import javaawtevent*;
import javaawtprint*;
public class TextPrint
{
public static void main(final String args[])
{
JFrame frame = new JFrame(打印测试);
framesetDefaultCloseOperation(JFrameEXIT_ON_CLOSE);
final JTextArea textArea = new JTextArea();
JScrollPane pane = new JScrollPane(textArea);
frameadd(pane BorderLayoutCENTER);
textAreasetText(打印内容\r\n可以分页! );
JButton button = new JButton(打印);
frameadd(button BorderLayoutSOUTH);
ActionListener listener = new ActionListener()
{
public void actionPerformed(ActionEvent e)
{
try
{
textAreaprint();
}
catch (PrinterException pe)
{
Systemerrprintln(打印失败);
}
}
};
buttonaddActionListener(listener);
framesetSize( );
framesetVisible(true);
}
}
图和图分别是打印对话框和设置对话框点击打印按钮后弹出如图的对话框
图 打印界面
图 设置对话框
虽然提供了打印设置对话框但我们并无法设置如页眉(角)等信息幸运的是print的一个重载为我们提供了这个功能下面是这个方法的参数
public boolean print(MessageFormat headerFormat
MessageFormat footerFormat
boolean showPrintDialog
PrintService service
PrintRequestAttributeSet attributes
boolean interactive)
增强的拖放功能
在Java SE 中的拖放功能得到了增强这主要表现在两个方面
·可以定制拖放模式
可以在拖放的过程中加入其它的辅助信息 首先需要通过JListJTable等控件的setDropMode()方法来设置一个拖动模式所有的控件都可以使用USER_SELECTION模式这个模式在以前的Java SE版本中就有这也是默认的拖放模式
JListJTable和Jlist都支持ON模式这个模式允许你将对象拖到其它项的上方而INSERT模式允许将一个对象插入在其它项之间而ON_OR_INSERT模式是前种模式的组合下面的代码将演示一个拖动的例子
import javaawt*;
import javaawtdatatransfer*;
import javaawtevent*;
import javaio*;
import javaxswing*;
import javaxswingtree*;
public class TestDrapDrop
{
public static void main(String args[])
{
JFrame f = new JFrame(拖放测试);
fsetDefaultCloseOperation(JFrameEXIT_ON_CLOSE);
JPanel top = new JPanel(new BorderLayout());
JLabel dragLabel = new JLabel(拖我:);
JTextField text = new JTextField();
textsetDragEnabled(true);
topadd(dragLabel BorderLayoutWEST);
topadd(text BorderLayoutCENTER);
fadd(top BorderLayoutNORTH);
final JTree tree = new JTree();
final DefaultTreeModel model = (DefaultTreeModel) treegetModel();
treesetTransferHandler(new TransferHandler()
{
public boolean canImport(TransferHandlerTransferSupport support)
{
if (!supportisDataFlavorSupported(DataFlavorstringFlavor) || !supportisDrop())
{
return false;
}
JTreeDropLocation dropLocation = (JTreeDropLocation) supporgetDropLocation();
return dropLocationgetPath() != null;
}
public boolean importData(TransferHandlerTransferSupport support)
{
if (!canImport(support))
{
return false;
}
JTreeDropLocation dropLocation = (JTreeDropLocation) supportgetDropLocation();
TreePath path = dropLocationgetPath();
Transferable transferable = supportgetTransferable();
String transferData;
try
{
transferData = (String) transferable getTransferData(DataFlavorstringFlavor);
}
catch (IOException e)
{
return false;
}
catch (UnsupportedFlavorException e)
{
return false;
}
int childIndex = dropLocationgetChildIndex();
if (childIndex == )
{
childIndex = modelgetChildCount(pathgetLastPathComponent());
}
DefaultMutableTreeNode newNode = new DefaultMutableTreeNode( transferData);
DefaultMutableTreeNode parentNode = (DefaultMutableTreeNode) pathgetLastPathComponent();
modelinsertNodeInto(newNode parentNode childIndex);
TreePath newPath = pathpathByAddingChild(newNode);
treemakeVisible(newPath);
treescrollRectToVisible(treegetPathBounds(newPath));
return true;
}
});
JScrollPane pane = new JScrollPane(tree);
fadd(pane BorderLayoutCENTER);
JPanel bottom = new JPanel();
JLabel comboLabel = new JLabel(DropMode);
String options[] ={ USE_SELECTION ON INSERT ON_OR_INSERT };
final DropMode mode[] =
{DropModeUSE_SELECTION DropModeON DropModeINSERT DropModeON_OR_INSERT };
final JComboBox combo = new JComboBox(options);
comboaddActionListener(new ActionListener()
{
public void actionPerformed(ActionEvent e)
{
int selectedIndex = combogetSelectedIndex();
treesetDropMode(mode[selectedIndex]);
}
});
bottomadd(comboLabel);
bottomadd(combo);
fadd(bottom BorderLayoutSOUTH);
fsetSize( );
fsetVisible(true);
}
图为拖动程序的运行界面在上面的文本框里输入相应的文本然后将其选择再拖动到下方的树中
图 拖动界面