Java SE 6之GUI:让界面更加绚丽(下)

本文为原创,如需转载,请注明作者和出处,谢谢!

Java SE 6之GUI:让界面更加绚丽(
上)



在上一篇中我介绍了Java SE 6在GUI上的部分改进。在这篇
文章中我接着介绍另外几种新的GUI功能。这些功能是:

1. 
带有排序和过滤功能的JTable。

2. 
增强的JTabbedPane组件

3. 
增强的打印功能

4. 
增强的拖放功能

一、带有排序和过滤功能的JTable

   
在Java SE 6中除了 java.awt被更新外,javax.swing同时也有了很大的改进。在 C/S程序中我们会经常使用到
“表”。如我们可以在查询数据库后将查询结果显示在表格中。在Java中显示表格使用的是JTable类。在以前的版本中,JTable只能简单地显示数据,并没有什么附加的处理功能,而在Java
SE 6中的JTable增加了排序和过滤功能。用户可以单击列头进行排序,也可以根据某一列来过滤表中的数
据。

为了使JTable可以对数据进行,必须将<
/span>RowSorter类和 JTable进行关联。 RowSorter是一个抽象类,它负责将
JTable中的数据映射成可排序的数据。在真正使用时,我们将直接使用 RowSorter的子类TableRowSorter。下面的代码显示了如何将 TableRowSorter类和JTable相关联。

TableModel model = new DefaultTableModel(rows, columns);
JTable table = new JTable(model);
RowSorter sorter = new TableRowSorter(model);
table.setRowSorter(sorter);

上面代码首先建立一个TableModel,然后将这个
TableModel的实例同时传递给了JTable和 RowSorter。下面是一个使用 JTable排序的简单的例子。

import javax.swing.*;
import javax.swing.table.*;
import java.awt.*;

public class TestSortedTable
{
    public static void main(String args[])
    {
        
JFrame frame = 
new JFrame("JTable的排序测试");
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
       
// 表格中显示的数据
        Object rows[][] =
        {
        
{ " 王明
", "中国", 44 },
        
{ " 姚明
", "中国", 25 },
        
{ " 赵子龙
", "西蜀", 1234 },
        
{ " 曹操
", "北魏", 2112 },
        
{ " Bill Gates
", "美国", 45 },
        
{ " Mike", "<
/span>英国", 33 } };
        String columns[] =
        
{ " 姓名
", "国籍", "年龄" };
        TableModel model = new DefaultTableModel(rows, columns);
        JTable table = new JTable(model);
        RowSorter<TableModel> sorter = new TableRowSorter<TableModel>(model);
        table.setRowSorter(sorter);
        JScrollPane pane = new JScrollPane(table);
        frame.add(pane, BorderLayout.CENTER);
        frame.setSize(300, 150);
        frame.setVisible(true);
    }
}

图1和图2
分别是按“姓名”进行升序和降序排列的显示结果。

图1
按“姓名”升序显示

图2 按“姓名”降序显示

   
图3显示的是按“年龄”进行降序排列。但我们发现一个奇怪的问题
,就是“年龄”字段并不是按数值类型进行排序的,而是按字符类型进行排序的。

 

图3 按年龄降序显示

出现这种情况是因为在默认情况下DefaultTableModal的
列是Object类型。而要想使JTable按数值进行排序,必须要覆盖
DefaultTableModal的getColumnClass方法。

TableModel model = new DefaultTableModel(rows, columns)
        {
            public Class getColumnClass(int column)
            {
                Class returnValue;
                if ((column >= 0) && (column < getColumnCount()))
                {
                    returnValue = getValueAt(0, column).getClass();
                }
                else
                {
                    returnValue = Object.class;
                }
                return returnValue;
            }
        };

 图4显示了按“年龄”进行排序的界面,看看,是不是按数值进行排序了。

图4 按数值类型进行排序

下面让我们来看看来何使用JTable进行过滤。我们可以
通过convertRowIndexToModel方法进行过滤。下面的代码加在一个按钮中添加事件代码调用JTable的过滤功能。

button.addActionListener(new ActionListener()
        {
            public void actionPerformed(ActionEvent e)
            {
                String text = filterText.getText();
                if (text.length() == 0)
                {
                    sorter.setRowFilter(null);
                }
                else
                {
                    sorter.setRowFilter
(RowFilter.regexFilter(text));
                }
            }
        });

 上面的代码并没有调用convertRowIndextoModel()方法,如果调用它,你就可以在
表中进行相应的操作。

在 JTable中通过抽象类 RowFilter类对行进行过滤。和排序不同,你可以不建立它们的子类,而使用这个抽象类的6个静态方法。

1.
   
andFilter

2.
       
dateFilter(RowFilter.ComparisonType
type, Date date, int...
indices)

3.
       
notFilter(RowFilter<M,I>
filter)

4.
       
numberFilter(RowFilter.ComparisonType
type, Number number, int...
indices)

5.
       
orFilter

6.
       
regexFilter(String
regex,
int... indices)

其中andFilter()、orFilter()以及 notFilter()方法的功能是将当前的过滤条件和其它的过滤条件进行组合。如在同时比较日期和数
值时需要将日期过滤和数值过滤进行组合。这些组合是非常简单的。

RowFilter的类型比较允许你进行4种关系的比较,等于
、不等于、大于或小于。我们可以通过指定某一列进行过滤,也可以对所有的列进行过滤。这其中最为有趣的也许是正则表达式过滤(regular expression filter,或简称为regex
filter)。使用这个过滤器可以对表中数据进行更高级的过滤。下面是实现一个简
单过滤器的代码。

import javax.swing.*;
import javax.swing.table.*;
import java.awt.*;
import java.awt.event.*;

public class TestFilter
{
    public static void main(String args[])
    {
        
JFrame frame = 
new JFrame("JTable的过滤测试");
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        Object rows[][] =
        {
        
{ "
王明
", "中国", 44 },
        
{ " 姚明
", "中国", 25 },
        
{ " 赵子龙
", "西蜀", 1234 },
        
{ " 曹操
", "北魏", 2112 },
        
{ " Bill Gates
", "美国", 45 },
        
{ " Mike", "<
/span>英国", 33 } };
        String columns[] =
        
{ " 姓名
", "国籍", "年龄" };
        TableModel model = new DefaultTableModel(rows, columns)
        {
            public Class getColumnClass(int column)
            {
                Class returnValue;
                if ((column >= 0) && (column < getColumnCount()))
                {
                    returnValue = getValueAt(0, column).getClass();
                }
                else
                {
                    returnValue = Object.class;
                }
                return returnValue;
            }
        };
        final JTable table = new JTable(model);
        final TableRowSorter<TableModel> sorter = new TableRowSorter<TableModel>(
                        model);
        table.setRowSorter(sorter);
        JScrollPane pane = new JScrollPane(table);
        frame.add(pane, BorderLayout.CENTER);
        JPanel panel = new JPanel(new BorderLayout());
        
JLabel label =  new JLabel("过滤");
        panel.add(label, BorderLayout.WEST);
        final JTextField filterText = new JTextField("");
        panel.add(filterText, BorderLayout.CENTER);
        frame.add(panel, BorderLayout.NORTH);
        
JButton button = 
new JButton("过滤");
        button.addActionListener(new ActionListener()
        {
            public void actionPerformed(ActionEvent e)
            {
                String text = filterText.getText();
                if (text.length() == 0)
                {
                    sorter.setRowFilter(null);
                }
                else
                {
                    sorter.setRowFilter
(RowFilter.regexFilter(text));
                }
            }
        });
        frame.add(button, BorderLayout.SOUTH);
        frame.setSize(300, 250);
        frame.setVisible(true);
    }
}

图5是上面程序的运行界面。

二、增强的JTabbedPane组件

JTabbedPane组件为我们提供了一种非常好的方法在窗体上显示很多的控件。我们可以将不同类别的控
件放到不同的Tab页上,然后通过需要点击相应的Tab页。在传统的 Tab页上只能防止文本的图标。而在 Java SE
6中使我们可以直接将控件放到 Tab上。我们可以通过 setTabComponentAt方法将控件放到Tab上。这个方法有两个参数,一个是Tab的索引,另一个是要放置的对象。

JTabbedPane pane = new JTabbedPane();
pane.setTabComponentAt(1, component);

在JTabbedPane控件中有3个常用的方法,setTabComponentAt(int
index, Component comp), getTabComponentAt
(int index)和indexOfTabComponent(Component)。最后一个方法将替换Tab上的控件。下面的代码是一个关于JTabbedPane控件的演示。

import javax.swing.*;
import javax.swing.table.*;
import java.awt.*;
import java.awt.event.*;

public class TestTabbedPane
{
    static void addIt(JTabbedPane tabbedPane, String text)
    {
        JLabel label = new JLabel(text);
        JButton button = new JButton(text);
        JPanel panel = new JPanel();
        panel.add(label);
        panel.add(button);
        tabbedPane.addTab(text, panel);
        if(text.equals
("tab4"))
            tabbedPane.setTabComponentAt(tabbedPane.getTabCount() - 1,
        
               
     new JTextField("插入了文本控件" ));
        else            
        tabbedPane.setTabComponentAt(tabbedPane.getTabCount() -
 1,
                        button);
    }
    public static void main(String args[])
    {
        
JFrame f = new JFrame(" JTabbedPane演示");
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        JTabbedPane tabbedPane = new JTabbedPane();
        addIt(tabbedPane, "tab1");
        addIt(tabbedPane, "tab2");
        addIt(tabbedPane, "tab3");
        addIt(tabbedPane, "tab4");
        addIt(tabbedPane, "tab5");
        f.add(tabbedPane, BorderLayout.CENTER);
        f.setSize(400, 200);
        f.setVisible(true);
    }
}

图6是显示界面,其中在Tab4上插入了一个文本控件,在Tab1至Tab5上各插入了一个
按钮控件。

图6 JTabbedPane演示

三、增强的打印功能

   
自从Java SE 5开始, Sun就对控件的打印功能进行了加强。如 JTextField、JTextArea等。在 Java
SE 6中Sun为打印增加了分页功能。我们只需要调用 JtextField或 JTextArea的print方法就可以调用打印对话框。下面是一段测试代
码。

import javax.swing.*;
import java.awt.*;
import java.awt.event.*;
import java.awt.print.*;

public class TextPrint
{
    public static void main(final String args[])
    {
        
JFrame frame = 
new JFrame("打印测试");
        frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        final JTextArea textArea = new JTextArea();
        JScrollPane pane = new JScrollPane(textArea);
        frame.add(pane, BorderLayout.CENTER);
        
textArea.setText("打印
内容/r/n可以分页!
" );
        
JButton button = 
new JButton("打印");
        frame.add(button, BorderLayout.SOUTH);
        ActionListener listener = new ActionListener()
        {
            public void actionPerformed(ActionEvent e)
            {
                try
                {
                    textArea.print();                       
                }
                catch (PrinterException pe)
                {
        
            System.err.println(
"打印失败");
                }
            }
        };
        button.addActionListener(listener);
        frame.setSize(250, 150);
        frame.setVisible(true);
}
}

图7和图8分别是打印对话框和设置对话框,
点击“打印“按钮后弹出如图8的对话框。


图7
打印界面

图8 设置对话框

   
虽然提供了打印设置对话框,但我们并无法设置如页眉(角)等信息,幸运的是print的一个重载为我们提供了这个功能。下面是这个方法的参数。

public boolean
print(MessageFormat
headerFormat,

MessageFormat
footerFormat,

boolean
showPrintDialog,

PrintService
service,

PrintRequestAttributeSet
attributes,

boolean
interactive)

四、增强的拖放功能

在 Java SE 6中的拖放功能得到了增强,这主要表现在两个
方面。

1.
     
可以定制拖放模式。

2.
       
可以在拖放的过程中加入其它的辅助信息。

   
首先需要通过JList、JTable等控件的
setDropMode()方法来设置一个拖动模式。所有的控件都可以使用 USER_SELECTION模式。这个模式在以前的Java SE版本中就有。这也是默认的拖放模式。

JList、JTable和Jlist都支持ON模式,这个模式允许你将对象拖到其它项的上方。而INSERT模式允许将
一个对象插入在其它项之间。而ON_OR_INSERT模式是前3种模式的组合。下面的代码将演
示一个拖动的例子。

import java.awt.*;
import java.awt.datatransfer.*;
import java.awt.event.*;
import java.io.*;
import javax.swing.*;
import javax.swing.tree.*;

public class TestDrapDrop
{
    public static void main(String args[])
    {
        
JFrame f = new JFrame(" 拖放测试" );
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        JPanel top = new JPanel(new BorderLayout());
        
JLabel dragLabel = new  JLabel("拖我:");
        JTextField text = new JTextField();
        text.setDragEnabled(true);
        top.add(dragLabel, BorderLayout.WEST);
        top.add(text, BorderLayout.CENTER);
        f.add(top, BorderLayout.NORTH);
        final JTree tree = new JTree();
        final DefaultTreeModel model = (DefaultTreeModel)
 tree.getModel();
        tree.setTransferHandler(new TransferHandler()
        {
            public boolean canImport
(TransferHandler.TransferSupport support)
            {
                if (!support.isDataFlavorSupported
(DataFlavor.stringFlavor)
                        || !support.isDrop())
                {
                    return false;
                }
                JTree.DropLocation dropLocation = (JTree.DropLocation) suppor
                        .getDropLocat
ion();
                return dropLocation.getPath() != null;
            }

            public boolean importData
(TransferHandler.TransferSupport support)
            {
                if (!canImport(support))
                {
                    return false;
                }
                JTree.DropLocation dropLocation = (JTree.DropLocation) support
                        .getDropLocat
ion();
                TreePath path = dropLocation.getPath();
                Transferable transferable = support.getTransferable();
                String transferData;
                try
                {
                    transferData = (String) transferable
                          &
nbsp; .getTransferData(DataFlavor.stringFlavor);
                }
                catch (IOException e)
                {
                    return false;
                }
                catch (UnsupportedFlavorException e)
                {
                    return false;
                }
                int childIndex = dropLocation.getChildIndex();
                if (childIndex == -1)
                {
                    childIndex = model.getChildCount(path
                          &
nbsp; .getLastPathComponent());
                }
                DefaultMutableTreeNode newNode = new DefaultMutableTreeNode(
                        transferData)
;
                DefaultMutableTreeNode parentNode = (DefaultMutableTreeNode) path
                        .getLastPathC
omponent();
                model.insertNodeInto
(newNode, parentNode, childIndex);
                TreePath newPath = path.pathByAddingChild(newNode);
                tree.makeVisible(newPath);
                tree.scrollRectToVisible(tree.getPathBounds(newPath));
                return true;
            }
        });
        JScrollPane pane = new JScrollPane(tree);
        f.add(pane, BorderLayout.CENTER);
        JPanel bottom = new JPanel();
        JLabel comboLabel = new JLabel("DropMode");
        String options[] =
        { "USE_SELECTION", "ON", "INSERT", "ON_OR_INSERT" };
        final DropMode mode[] =
        { DropMode.USE_SELECTION, DropMode.ON, DropMode.INSERT,
                DropMode.ON_OR_INSERT };
        final JComboBox combo = new JComboBox(options);
        combo.addActionListener(new ActionListener()
        {
            public void actionPerformed(ActionEvent e)
            {
                int selectedIndex = combo.getSelectedIndex();
                tree.setDropMode(mode[selectedIndex]);
            }
        });
        bottom.add(comboLabel);
        bottom.add(combo);
        f.add(bottom, BorderLayout.SOUTH);
        f.setSize(300, 400);
        f.setVisible(true);
    }
}

图 9为拖动程序的运行界面。在上面的文本框里输入相应的
文本,然后将其选择再拖动到下方的树中。

图 9 拖动界面

国内最棒的Google Android技术社区(eoeandroid),欢迎访问!

《银河系列原创教程》发布

《Java Web开发速学宝典》出版,欢迎定购

时间: 2024-08-31 03:07:11

Java SE 6之GUI:让界面更加绚丽(下)的相关文章

Java SE 6之GUI:让界面更加绚丽(上)

本文为原创,如需转载,请注明作者和出处,谢谢! 本文曾发表于天极网:http://dev.yesky.com/185/2690685.shtml Java SE 6之GUI:让界面更加绚丽(下) Java是一种非常强大的编程语言,但Java的GUI能力一直是它的软肋.虽然Java提供2种图形解决方案,AWT和Swing.但这2种图形库的功能十分有限,使用它们并不能带来更多的用户体验.而且AWT和Swing的更新速度很慢.一直到Java SE 5也没什么明显的变化.幸好Java SE 6终于为它们

让界面更加绚丽 Java SE 6.0 GUI体验

Java是一种非常强大的编程语言,但Java的GUI能力一直是它的软肋.虽然Java提供2种图形解决方案,AWT和Swing.但这2种图形库的功能十分有限,使用它们并不能带来更多的用户体验.而且AWT和Swing的更新速度很慢.一直到Java SE 5也没什么明显的变化.幸好Java SE 6终于为它们增加了更强大的功能.有了这些功能,使用AWT和Swing设计的用户界面可以更贴近用户.为了使读者能够更早地体验Java SE 6 GUI的美丽,本文将就Java SE 6 GUI提供的主要功能进行

用Java SE 6.0实现高质量桌面集成开发

提要:本文将以具体的实例向你展示Java SE 6在桌面开发中所提供的一系列优秀特征. 一. 引言 随着Java SE 6 Beta的发行,Java开发者再也不需要自己去实现Java本机接口(JNI)便可以在其应用程序中加入桌面产品所具有的特征.这些桌面集成特征现在已成为内核的一个组成部分. 最新的Java标准版本6.0(代码名为Mustang)将在功能上进行一系列的改进--开发者们能够方便地用Java处理安全,JMX,操作系统文件,国际化和桌面开发等一系列问题.Sun在开发这个版本的过程中表现

Java SE 6在Solaris的可观察性特征分析

Java平台标准版(Java SE)6,代码名为"Mustang",是最新的Java SE发行版本(正在开发中).Java SE 6源码和二进制代码都可以在www.Java.net上下载.Java SE 6平台要到2006年秋天才能正式发布.不过,现在你可以下载试用Java SE 6 Beta版. Java SE 6平台中提供了多种可观察性(observability)工具,这其中的许多工具都可在系统中运行,而这些工具中的只有极少数被用于挂起进程或核心复制处理.因此,在本文中,我们将分

Java SE 6中的AWT模态增强功能

一. 引言 模态一直是我最喜欢的话题之一,至少因为一直存在大量的感兴趣的客户的有关于这方面内容的回馈信息.作为一个Java SE测试工程师,这正是驱动我工作的动力-它使我一直处于忙碌之中,而且还为我的工作带来快乐.另一方面,这也说明了模态是客户端最广为使用的特征之一,不管是使用Swing还是使用AWT进行开发. 在Java SE 6(代码名为Mustang)中,我的工作变得更有意思--在模态方面进行大量的改进.这些改进将为应用程序开发者提供更大的灵活性--在设计可能用到模态对话框行为的程序的过程

Java SE 6 新特性: HTTP 增强

2006 年底,Sun 公司发布了 Java Standard Edition 6(Java SE 6)的最终正式版,代号 Mustang(野马).跟 Tiger(Java SE 5)相比,Mustang 在性能方面有了不错的提升.与 Tiger 在 API 库方面的大幅度加强相比,虽然 Mustang 在 API 库方面的新特性显得不太多,但是也提供了许多实用和方便的功能:在脚本,WebService,XML,编译器 API,数据库,JMX,网络和 Instrumentation 方面都有不错

Java SE 8:标准库增强

Lambda表达式是Java SE 8的核心功能,大部分的改进都围绕lambda表达式展开.(Jigsaw项目已经被推迟到Java SE 9.)关于lambda表达式的内容,已经在上一篇文章中进行了说明.这篇文章主要介绍Java SE 8中包含的其他Java标准库的增强. 并行排序 随着多核CPU的流行,Java平台的标准库实现也尽可能利用底层硬件平台的能力来提高性能.Java SE 7中引入了Fork/Join框架作为一个轻量级的并行任务执行引擎.Java SE 8把Fork/Join框架用到

Java SE 8: Lambda表达式

Java SE 8在6月13的版本中已经完全了全部的功能.在这些新的功能中,lambda表达式是推动该版本发布 的最重要新特性.因为Java第一次尝试引入函数式编程的相关内容.社区对于lambda表达式也期待已久. Lambda表达式的相关内容在JSR 335中定义,本文的内容基于最新的规范和JDK 8 Build b94. 开发环境使用 的是Eclipse. Lambda表达式 要理解lambda表达式,首先要了解的是函数式接口(functional interface).简单来说,函数式接口

Java SE 6 update 10 Beta简介

不要被这个普普通通的名字所迷惑,即将发布的 Java SE6 update10(当前发布的是试用版)与之前 的更新版本存在很大的差异.与之前发布的 java 编程语言更新相比,Java SE 6u10 加入了更多的特性 和功能,从而推动包络(Envelope)的发展. 严格地说,"Java更新发行版"只是对版本号的第四位(比如,对版本 1.6.0_10 的 10 )进行了改 动.与大多数发行版不同,更新发行版本不允许添加.删除或者更改任何公共 API 函数,这样也将更新 发行局限于对