• 图形用户界面(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
  • 相关阅读:
    day08
    day07
    day06
    day06
    day05
    第三次作业
    第三次作业
    第二次作业
    java 数字和日期处理
    jsp文件导包
  • 原文地址:https://www.cnblogs.com/penghuster/p/4861591.html
Copyright © 2020-2023  润新知