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(
300150);
        frame.setVisible(
true);
    }
}
1 和图 2  分别是按“姓名”进行升序和降序排列的显示结果。


按“姓名”升序显示



按“姓名”降序显示
     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 显示了按“年龄”进行排序的界面,看看,是不是按数值进行排序了。


按数值类型进行排序
下面让我们来看看来何使用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(
300250);
        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 的对话框。

打印界面

设置对话框
     虽然提供了打印设置对话框,但我们并无法设置如页眉(角)等信息,幸运的是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 为拖动程序的运行界面。在上面的文本框里输入相应的文本,然后将其选择再拖动到下方的树中。

图 拖动界面





 本文转自 androidguy 51CTO博客,原文链接: http://blog.51cto.com/androidguy/216470 ,如需转载请自行联系原作者
相关文章
|
2月前
|
前端开发 Oracle Java
Java中的GUI应用开发技术选型
Java中的GUI应用开发技术选型
|
1月前
|
存储 数据可视化 Java
【Java】Java swing 民宿管理系统 GUI(源码+可视化界面)【独一无二】
【Java】Java swing 民宿管理系统 GUI(源码+可视化界面)【独一无二】
|
2月前
|
Java Apache Android开发
Java中的GUI界面设计与用户体验优化
Java中的GUI界面设计与用户体验优化
|
3月前
|
前端开发 Java 开发工具
Java GUI编程:跨平台应用的设计与开发
Java GUI编程:跨平台应用的设计与开发
|
2月前
|
前端开发 Java 开发工具
Java GUI编程:跨平台应用的设计与开发
Java GUI编程:跨平台应用的设计与开发
|
2月前
|
Java 数据安全/隐私保护 容器
Java详解:GUI容器组件 | 功能组件
Java详解:GUI容器组件 | 功能组件
|
2月前
|
Java 容器
Java详解:GUI图形用户界面设计—容器组件及面板布局方式
Java详解:GUI图形用户界面设计—容器组件及面板布局方式
|
9天前
|
监控 Java 调度
【Java学习】多线程&JUC万字超详解
本文详细介绍了多线程的概念和三种实现方式,还有一些常见的成员方法,CPU的调动方式,多线程的生命周期,还有线程安全问题,锁和死锁的概念,以及等待唤醒机制,阻塞队列,多线程的六种状态,线程池等
67 6
【Java学习】多线程&JUC万字超详解
|
2天前
|
Java 调度 开发者
Java并发编程:深入理解线程池
在Java的世界中,线程池是提升应用性能、实现高效并发处理的关键工具。本文将深入浅出地介绍线程池的核心概念、工作原理以及如何在实际应用中有效利用线程池来优化资源管理和任务调度。通过本文的学习,读者能够掌握线程池的基本使用技巧,并理解其背后的设计哲学。
|
2天前
|
缓存 监控 Java
Java中的并发编程:理解并应用线程池
在Java的并发编程中,线程池是提高应用程序性能的关键工具。本文将深入探讨如何有效利用线程池来管理资源、提升效率和简化代码结构。我们将从基础概念出发,逐步介绍线程池的配置、使用场景以及最佳实践,帮助开发者更好地掌握并发编程的核心技巧。