图形用户界面(graphical user interface)

1 java中提供的类库

1.1 定义

  • AWT(abstract windows toolkit)抽象窗口工具包:提供了与本地图形界面进行交互的接口,AWT中提供的图形函数与操作系统的图形函数有着对应关系。(为了实现java的平台通用性,AWT牺牲了部分功能,整合所有操作系统的共有图形函数,形成了AWT工具包;由于依赖于本地函数,AWT控件也叫重量级控件)
  • Swing是在AWT的基础上构建的一套新的图形函数库,提供了所有AWT的功能,并用100%的java代码对AWT进行了扩充。(平台通用性更好,不使用本地图形操作函数,Swing控件也叫轻量级控件)
  • AWT与Swing区别:AWT速度快,Swing速度慢,但移植性更好

1.2 java包

  • java.awt包:Frame Button Label  TextField TextArea Panel
  • javax.swing包:JFrame JButton JLabel JTextField JTextArea JPanel

1.3 组件分类

  java中构成用户图形界面的各个元素,统称为组件(Component)

  Component又分为容器(Container)和非容器两大类

  容器又分为顶层容器和非顶层容器

1.3.1 Component类

  • 是所有图形化组件和容器的抽象父类,其中定义了每个容器和组件都可能用到的通用方法
  • getBounds() , getSize(), getLocation(), getHeight(), getWidth()
  • setVisible(), setEnabled(),setBackground(), setForeground()
  • getGraphics()
  •  requestFocus()

1.3.2 Swing容器

  • 顶层容器:JFrame, JDialog, JApplet
  • 非顶层容器:JComponent

2 实现图形界面

2.1 实现图形界面的三步曲:

  • 创建组件(Component):创建组成界面的各种元素,如按钮、文本框等。
  • 指定布局(Layout):根据具体需要排列它们的位置关系。
  • 响应事件(Event) : 定义图形用户界面的事件和各界面元素对不同事件的响应, 从而实现图形用户界面与用户 的交互功能。

2.2 示例

import java.awt.*;

import javax.swing.*;


public class ButtonDemo extends JFrame {
    JButton btn1 = new JButton("Jbutton1");
    JButton btn2 = new JButton("dddd");
    JTextField txt = new JTextField(20);
    public ButtonDemo(){
        super("test button");
        btn1.setToolTipText("显示点选按钮");
        btn2.setIcon(new ImageIcon("cupHJbutton.gif"));
        setLayout(new FlowLayout());
        getContentPane().add(btn1);
        getContentPane().add(btn2);
        getContentPane().add(txt);
        setSize(400,300);
        setDefaultCloseOperation(EXIT_ON_CLOSE);
        
        btn1.addActionListener((e)->{
            String  name = ((JButton)(e.getSource())).getText();
            txt.setText(name + " Pressed");
        });
        
        btn2.addActionListener((e)->{
            String  name = ((JButton)(e.getSource())).getText();
            txt.setText(name + " Pressed");
        });
    }

    public static void main(String[] args) {
        // TODO Auto-generated method stub
        new ButtonDemo().setVisible(true);
    }

}
View Code

3 布局管理

3.1 Java.awt包

  常用的三种:FlowLayout BorderLayout GridLayout,还有CardLayout, GridBagLayout等

3.1 FlowLayout 

  •  对组件逐行定位,行内从左到右,一行排满后换行
  • 默认对齐方式为居中对齐,水平和竖直间距为缺省值为5
  • 不改变组件的大小,按组件原有尺寸显示组件
  • FlowLayout是Panel类的默认布局管理器

3.2 BorderLayout 布局管理器

  • BorderLayout将整个容器的布局划分成东、西、南、北、中五个区域,组件只能被添加 到指定的区域
  • 如不指定组件的加入部位,则默认加入到Center区域
  • 每个区域只能加入一个组件,如加入多个,则先前加入的组件会被遗弃
  • BorderLayout是Frame类的默认布局管理器

3.3 GridLayout 布局管理器

  • GridLayout型布局管理器将布局划分成规则的矩形网格,每个单元格区域大小相等.
  • 组件被添加到每个单元格中,先从左到右添满一行后换行,再从上到下.
  • 在GridLayout构造方法中指定分割的行数和列数.

3.4 CardLayout 布局管理器

  • CardLayout布局管理器能够帮助用户处理两个以至更多的成员共享同 一显示空间,就好象一叠卡片摞在一起。

3.5 默认的布局管理器

4 事件处理

4.1 定义

  • 事件event:鼠标,键盘,布局改变等各种操作
  • 事件监听器event Listener:对事件作出相应响应的程序,是AWTEventListener的子接口

4.2 事件适配器Adapter-----简化Listener

  • 如WindowListener有7个方法,即使一些方法不做任何事情也得书写完全
  • 在适配器类中,实现了相应监听接口的所有方法,但不做任何事情
  • 在extends事件适配器类时,只需要override自己所需要的方法即可

4.3 事件处理的步骤

  • 创建监听器,并实现监听功能
  • 为监听器注册待监听对象

4.4 创建监听器的6种方法

4.4.1 通过类本身实现监听器

import java.awt.*;
import java.awt.event.*;

import javax.swing.*;
public class TestActionEventByAnonymous extends JFrame implements ActionListener{
    JTextField txt = new JTextField( 20);
    JPanel pnl = new JPanel();
    JButton b1 = new JButton("1");
    JButton b2 = new JButton("2");
    JButton b3 = new JButton("3");
    JButton b4 = new JButton("4");
    
    public void actionPerformed(ActionEvent e){
        if(e.getSource() == b1)
            txt.setText("b1" + " Pressed");
        else if(e.getSource() == b2)
            txt.setText("b2" + " Pressed");
    }
    public TestActionEventByAnonymous(){
        super("Nested Container");
        
        pnl.setLayout(new GridLayout(2,2));
        pnl.add(b1);         pnl.add(b2);
        pnl.add(b3);        pnl.add(b4);
        
        add(txt, BorderLayout.NORTH);
        add(pnl, BorderLayout.CENTER);
        b1.addActionListener(this);
        b2.addActionListener(this);
        
        setSize(200, 120);
        setDefaultCloseOperation(EXIT_ON_CLOSE);
        setVisible(true);      
    }
    public static void main(String args[]) {
        new TestActionEventByAnonymous();
    }
}
View Code

4.4.2 通过外部类实现监听器

import java.awt.*;
import java.awt.event.*;

import javax.swing.*;
public class TestActionEventByAnonymous extends JFrame {
    JTextField txt = new JTextField( 20);
    JPanel pnl = new JPanel();
    JButton b1 = new JButton("1");
    JButton b2 = new JButton("2");
    JButton b3 = new JButton("3");
    JButton b4 = new JButton("4");
    
    public TestActionEventByAnonymous(){
        super("Nested Container");
        
        pnl.setLayout(new GridLayout(2,2));
        pnl.add(b1);         pnl.add(b2);
        pnl.add(b3);        pnl.add(b4);
        
        add(txt, BorderLayout.NORTH);
        add(pnl, BorderLayout.CENTER);
        ButtonEventListener btnListener = new ButtonEventListener(txt);
        ButtonEventListener btnListener2 = new ButtonEventListener(txt);
        b1.addActionListener(btnListener);
        b2.addActionListener(btnListener2);
        
        setSize(200, 120);
        setDefaultCloseOperation(EXIT_ON_CLOSE);
        setVisible(true);      
    }
    public static void main(String args[]) {
        new TestActionEventByAnonymous();
    }
}

//外部类DialogEventListener,实现ActionListener接口
class ButtonEventListener implements ActionListener {
    JTextField txt;
    public ButtonEventListener(JTextField txt){
        this.txt = txt;
    }
    @Override
    public void actionPerformed(ActionEvent e) {
        //创建JDialog窗口对象
        String name = ((JButton)e.getSource()).getText();
            txt.setText(name + " Pressed");
    }

}
View Code

4.4.3 通过内部类实现监听器

import java.awt.*;
import java.awt.event.*;

import javax.swing.*;
public class TestActionEventByAnonymous extends JFrame {
    JTextField txt = new JTextField( 20);
    JPanel pnl = new JPanel();
    JButton b1 = new JButton("1");
    JButton b2 = new JButton("2");
    JButton b3 = new JButton("3");
    JButton b4 = new JButton("4");
    
    //外部类DialogEventListener,实现ActionListener接口
    class ButtonEventListener implements ActionListener {
        @Override
        public void actionPerformed(ActionEvent e) {
            //创建JDialog窗口对象
            String name = ((JButton)e.getSource()).getText();
                txt.setText(name + " Pressed");
        }

    }
    
    public TestActionEventByAnonymous(){
        super("Nested Container");
        
        pnl.setLayout(new GridLayout(2,2));
        pnl.add(b1);         pnl.add(b2);
        pnl.add(b3);        pnl.add(b4);
        
        add(txt, BorderLayout.NORTH);
        add(pnl, BorderLayout.CENTER);
        ButtonEventListener btnListener = new ButtonEventListener();
        b1.addActionListener(btnListener);
        b2.addActionListener(btnListener);
        
        setSize(200, 120);
        setDefaultCloseOperation(EXIT_ON_CLOSE);
        setVisible(true);      
    }
    public static void main(String args[]) {
        new TestActionEventByAnonymous();
    }
}
View Code

4.4.4 通过局部类实现监听器

import java.awt.*;
import java.awt.event.*;

import javax.swing.*;
public class TestActionEventByAnonymous extends JFrame {
    JTextField txt = new JTextField( 20);
    JPanel pnl = new JPanel();
    JButton b1 = new JButton("1");
    JButton b2 = new JButton("2");
    JButton b3 = new JButton("3");
    JButton b4 = new JButton("4");
    
    
    public TestActionEventByAnonymous(){
        super("Nested Container");
        
        pnl.setLayout(new GridLayout(2,2));
        pnl.add(b1);         pnl.add(b2);
        pnl.add(b3);        pnl.add(b4);
        
        add(txt, BorderLayout.NORTH);
        add(pnl, BorderLayout.CENTER);
        //局部类DialogEventListener,实现ActionListener接口
        class ButtonEventListener implements ActionListener {
            @Override
            public void actionPerformed(ActionEvent e) {
                //创建JDialog窗口对象
                String name = ((JButton)e.getSource()).getText();
                    txt.setText(name + " Pressed");
            }

        }
        ButtonEventListener btnListener = new ButtonEventListener();
        b1.addActionListener(btnListener);
        b2.addActionListener(btnListener);
        
        setSize(200, 120);
        setDefaultCloseOperation(EXIT_ON_CLOSE);
        setVisible(true);      
    }
    public static void main(String args[]) {
        new TestActionEventByAnonymous();
    }
}
View Code

4.4.5 通过匿名类实现监听器

import java.awt.*;
import java.awt.event.*;

import javax.swing.*;
public class TestActionEventByAnonymous extends JFrame {
    JTextField txt = new JTextField( 20);
    JPanel pnl = new JPanel();
    JButton b1 = new JButton("1");
    JButton b2 = new JButton("2");
    JButton b3 = new JButton("3");
    JButton b4 = new JButton("4");
    
    
    public TestActionEventByAnonymous(){
        super("Nested Container");
        
        pnl.setLayout(new GridLayout(2,2));
        pnl.add(b1);         pnl.add(b2);
        pnl.add(b3);        pnl.add(b4);
        
        add(txt, BorderLayout.NORTH);
        add(pnl, BorderLayout.CENTER);
        //局部类DialogEventListener,实现ActionListener接口
        class ButtonEventListener implements ActionListener {
            @Override
            public void actionPerformed(ActionEvent e) {
                //创建JDialog窗口对象
                String name = ((JButton)e.getSource()).getText();
                    txt.setText(name + " Pressed");
            }

        }
        b1.addActionListener(new ActionListener(){
                public void actionPerformed(ActionEvent e) {
                    txt.setText("b1" + " Pressed");
                }
            });
        b2.addActionListener(new ActionListener(){
                public void actionPerformed(ActionEvent e) {
                    txt.setText("b2" + " Pressed");
                }
            });
        setSize(200, 120);
        setDefaultCloseOperation(EXIT_ON_CLOSE);
        setVisible(true);      
    }
    public static void main(String args[]) {
        new TestActionEventByAnonymous();
    }
}
View Code

4.4.6 通过lambda表达式实现监听器

import java.awt.*;
import java.awt.event.*;

import javax.swing.*;
public class TestActionEventByAnonymous extends JFrame {
    JTextField txt = new JTextField( 20);
    JPanel pnl = new JPanel();
    JButton b1 = new JButton("1");
    JButton b2 = new JButton("2");
    JButton b3 = new JButton("3");
    JButton b4 = new JButton("4");
    
    
    public TestActionEventByAnonymous(){
        super("Nested Container");
        
        pnl.setLayout(new GridLayout(2,2));
        pnl.add(b1);         pnl.add(b2);
        pnl.add(b3);        pnl.add(b4);
        
        add(txt, BorderLayout.NORTH);
        add(pnl, BorderLayout.CENTER);
        //局部类DialogEventListener,实现ActionListener接口
        class ButtonEventListener implements ActionListener {
            @Override
            public void actionPerformed(ActionEvent e) {
                //创建JDialog窗口对象
                String name = ((JButton)e.getSource()).getText();
                    txt.setText(name + " Pressed");
            }

        }
        b1.addActionListener(e->{
            txt.setText("b1" + " Pressed");
            });
        b2.addActionListener(e->{
            txt.setText("b2" + " Pressed");
            });
        setSize(200, 120);
        setDefaultCloseOperation(EXIT_ON_CLOSE);
        setVisible(true);      
    }
    public static void main(String args[]) {
        new TestActionEventByAnonymous();
    }
}
View Code

4.4.7 总结

  • 类自身实现监听器的方法通过implements多个接口,可以实现创建多个监听器
  • 内部类/外部类均能实现一个监听器监听多个对象,或创建多个监听器,但外部类处理对象不方便
  • 局部类不能够在函数体外创建监听器,但是在函数体内能够监听多个对象,并能够创建多个监听器
  • 匿名类对于多个监听器监听同一个对象
  • lambda表达式只适用于仅有一个接口的监听器

4.5 事件与线程

  • 线程中,如果要更新界面,要放到the event dispatching thread ,即要调用 SwingUtilities.invokeLater()方法

5 应用示例

简单文本编辑器

import java.awt.*;
import java.awt.event.*;
import java.io.*;
import java.util.logging.*;

import javax.swing.*;



public class TextEditors  {
    
    public static void main( String [] args){

        javax.swing.SwingUtilities.invokeLater(new Runnable() {
            @Override
            public void run() {
                TextDAL dal = new FileTextDAL();
                TextEditorFrame f = new TextEditorFrame(dal);
                f.setTitle( "简单的编辑器");
                f.setSize( 800, 600 );
                f.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
                f.setVisible(true);
            }
        });
    }

}

class TextEditorFrame extends JFrame{
    JTextPane text = new JTextPane();
    JFileChooser fileChooser = new JFileChooser();
    JColorChooser colorChoser = new JColorChooser();
    JDialog about = new JDialog();
    JMenuBar menuBar = new JMenuBar();
    
    TextDAL dal = null;
    File file = null;
    Color color = Color.BLACK;
    
    TextEditorFrame(TextDAL dal){
        this.dal = dal;
        initTextPane();
        initMenu();
        initAboutDialog();
        initToolBar();
    }
    
    void initTextPane(){
        getContentPane().add(new JScrollPane(text));
    } 
    
    JMenu[] menus = new JMenu[]{new JMenu("File"),
            new JMenu("Edit"), new JMenu("Help")
    };
    JMenuItem[][] menuItems = new JMenuItem[][]{
            {new JMenuItem("New"), new JMenuItem("Open..."), new JMenuItem("Save..."),
                new JMenuItem("Exit")},
            {new JMenuItem("Copy"), new JMenuItem("Paste"),    new JMenuItem("Cut"), 
                    new JMenuItem("Color...")},
            {new JMenuItem("About")}
    };
    
    void initMenu(){
        for(int i=0; i<menus.length; i++){
            menuBar.add(menus[i]);
            for(int j=0; j<menuItems[i].length; j++){
                menus[i].add(menuItems[i][j]);
                menuItems[i][j].addActionListener(action);
            }
        }
        this.setJMenuBar( menuBar );
    }
    
    ActionListener action = new ActionListener(){
        public void actionPerformed(ActionEvent evt){
            JMenuItem mi = (JMenuItem)evt.getSource();
            String id = mi.getText();
            if(id.equals("New")){
                text.setText("");
                file = null;
            }else if(id.equals("Open...")){
                if(file != null) fileChooser.setSelectedFile(file);
                int returnVal = fileChooser.showOpenDialog(TextEditorFrame.this);
                if(returnVal == JFileChooser.APPROVE_OPTION){
                    file = fileChooser.getSelectedFile();
                    openFile();
                }
            }else if(id.equals("Save...")){
                if(file != null) fileChooser.setSelectedFile(file);
                int returnVal = fileChooser.showSaveDialog(TextEditorFrame.this);
                if(returnVal == JFileChooser.APPROVE_OPTION) {
                    file = fileChooser.getSelectedFile();
                    saveFile();    
                }
            }else if( id.equals("Exit")){
                System.exit(0);
            }else if( id.equals("Cut")){
                text.cut();
            }else if( id.equals("Copy")){
                text.copy();
            }else if( id.equals("Paste")){
                text.paste();
            }else if( id.equals("Color...")){
                color = JColorChooser.showDialog(TextEditorFrame.this, "", color );
                text.setForeground(color);
            }else if( id.equals("About")){
                about.setSize(100,50);
                about.setVisible(true);
            }
        }
    };
    
    void saveFile(){ //保存文件,将字符写入文件
        String content = text.getText();
        dal.save(file, content);
    }
    void openFile(){ //读入文件,并将字符置入文本框中
        String content = dal.read(file);
        text.setText(content);
    }
    
    void initAboutDialog(){
        about.getContentPane().add( new JLabel("简单编辑器 V1.0") );
        about.setModal( true );  
        about.setSize(100,50 );
    }
    
    JToolBar toolBar = new JToolBar();
    JButton[] buttons = new JButton[]{
        new JButton("", new ImageIcon("copy.jpg")),
        new JButton("", new ImageIcon("cut.jpg")),
        new JButton("", new ImageIcon("paste.jpg")),
    };
    void initToolBar(){
        for(int i=0; i<buttons.length; i++)
            toolBar.add(buttons[i]);
        buttons[0].setToolTipText( "copy" );
        buttons[0].addActionListener( new ActionListener(){
            public void actionPerformed( ActionEvent e ){
                text.copy();
            }
        });
        buttons[1].setToolTipText( "cut" );
        buttons[1].addActionListener( new ActionListener(){
            public void actionPerformed( ActionEvent e ){
                text.cut();
            }
        });
        buttons[2].setToolTipText( "paste" );
        buttons[2].addActionListener( new ActionListener(){
            public void actionPerformed( ActionEvent e ){
                text.paste();
            }
        });
        this.getContentPane().add( toolBar, BorderLayout.NORTH );
        toolBar.setRollover(true);
    }
}


//------------- 关于数据存取、关于日志 ---------------
interface TextDAL {
  String read(File file);
  void save(File file, String text);
}


class FileTextDAL implements TextDAL {

  @Override
  public String read(File file) {
        logger.log(Level.INFO, "read", "read..." + file.getPath());

        try{
            FileReader fr = new FileReader( file );
            int len = (int) file.length();
            char [] buffer = new char[len];
            fr.read( buffer, 0, len );
            fr.close();
            return new String( buffer );
        }catch(Exception e ){ 
            e.printStackTrace(); 
          logger.log(Level.SEVERE, null, e);
        }
        return "";
    }

    @Override
  public void save(File file, String text) {
        logger.log(Level.INFO, "save", "save..." + file.getPath());
        try{
            FileWriter fw = new FileWriter( file );
            fw.write( text );
            fw.close();
        }catch(Exception ex ){ 
            ex.printStackTrace(); 
          logger.log(Level.SEVERE, null, ex);
        }
    }

    //加点日志处理
    Logger logger = Logger.getLogger( FileTextDAL.class.getName());
    {
        try{
            FileHandler handler = new FileHandler("TextEditorApp2.log");//可以用 %h/xxxx.log表示在用户主目录下
            handler.setFormatter( new SimpleFormatter());
            logger.addHandler(handler);
        }catch(IOException ex){}
    }

}
View Code

 

posted on 2015-10-11 10:02  新手酱油  阅读(2625)  评论(0编辑  收藏  举报

导航