Cyh的博客

Email:kissyan4916@163.com
posts - 26, comments - 19, trackbacks - 0, articles - 220

网络编程>>Web浏览器

Posted on 2009-12-12 15:17 啥都写点 阅读(469) 评论(0)  编辑  收藏 所属分类: J2SE
     本例实现了一个Web浏览器,类似于Windows的Internet Explorer,但是功能相对简单一些,能显示HTML网页、前进、后退、刷新、访问主页和新建窗口,当所有窗口关闭时,程序结束。

 javax.swing.JEditorPane文本控件支持纯文本、HTML文本、RTF文本的显示,可以用它来显示HTML网页。
 JEditorPane的setPage方法加载URL,当加载完毕后,触发PropertyChangeListener事件,通过addPropertyChangeListener方法为JEditorPane注册属性改变事件处理器。
 为了使得网页上的超链接生效,必须通过addHyperlinkListener为JEditorPane注册超链接事件处理器。


/**---------------------------------WebBrowser.java----------------------------------------*/

package book.net.url;

import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.io.File;
import java.io.IOException;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.ArrayList;

/**
 * 实现一个简单的Web浏览器,支持HTML和HTM页面的显示。使用了JEditorPane组件
 *
*/

public class WebBrowser extends JFrame implements HyperlinkListener,
        PropertyChangeListener 
{
    
    
/**下面是使用的Swing组件**/
    
    
// 显示HTML的面板
    JEditorPane textPane; 
    
// 最底下的状态栏
    JLabel messageLine; 
    
// 网址URL输入栏
    JTextField urlField;
    
// 文件选择器,打开本地文件时用
    JFileChooser fileChooser;
    
    
// 后退和前进 按钮
    JButton backButton;
    JButton forwardButton;

    
// 保存历史记录的列表
    java.util.List history = new ArrayList(); 
    
// 当前页面的在历史记录列表中位置
    int currentHistoryPage = -1;  
    
// 当历史记录超过MAX_HISTORY时,清除旧的历史
    public static final int MAX_HISTORY = 50;

    
// 当前已经打开的浏览器窗口数
    static int numBrowserWindows = 0;
    
// 标识当所有浏览器窗口都被关闭时,是否退出应用程序
    static boolean exitWhenLastWindowClosed = false;

    
// 默认的主页
    String home = "http://www.google.com";

    
/**
     * 构造函数
     
*/

    
public WebBrowser() {
        
super("WebBrowser"); 

        
// 新建显示HTML的面板,并设置它不可编辑
        textPane = new JEditorPane(); 
        textPane.setEditable(
false); 

        
// 注册事件处理器,用于超连接事件。
        textPane.addHyperlinkListener(this);
        
// 注册事件处理器,用于处理属性改变事件。当页面加载结束时,触发该事件
        textPane.addPropertyChangeListener(this);

        
// 将HTML显示面板放入主窗口,居中显示
        this.getContentPane().add(new JScrollPane(textPane),
                                  BorderLayout.CENTER);

        
// 创建状态栏标签,并放在主窗口底部
        messageLine = new JLabel(" ");
        
this.getContentPane().add(messageLine, BorderLayout.SOUTH);

        
// 初始化菜单和工具栏
        this.initMenu();
        
this.initToolbar();
        
        
// 将当前打开窗口数增加1
        WebBrowser.numBrowserWindows++;
        
        
// 当关闭窗口时,调用close方法处理
        this.addWindowListener(new WindowAdapter() {
            
public void windowClosing(WindowEvent e) {
                close();
            }

        }
);
    }

    
    
/**
     * 初始化菜单栏
     
*/

    
private void initMenu(){
        
        
// 文件菜单,下面有四个菜单项:新建、打开、关闭窗口、退出
        JMenu fileMenu = new JMenu("文件");
        fileMenu.setMnemonic(
'F');
        JMenuItem newMenuItem 
= new JMenuItem("新建");
        newMenuItem.setMnemonic(
'N');
        
// 当“新建”时打开一个浏览器窗口
        newMenuItem.addActionListener(new ActionListener() {
            
public void actionPerformed(ActionEvent e) {
                newBrowser();
            }

        }
);
        
        JMenuItem openMenuItem 
= new JMenuItem("打开");
        openMenuItem.setMnemonic(
'O');
        
// 当“打开”是打开文件选择器,选择待打开的文件
        openMenuItem.addActionListener(new ActionListener() {
            
public void actionPerformed(ActionEvent e) {
                openLocalPage();
            }

        }
);
        
        JMenuItem closeMenuItem 
= new JMenuItem("关闭窗口");
        closeMenuItem.setMnemonic(
'C');
        
// 当“关闭窗口”时关闭当前窗口
        closeMenuItem.addActionListener(new ActionListener() {
            
public void actionPerformed(ActionEvent e) {
                close();
            }

        }
);
        
        JMenuItem exitMenuItem 
= new JMenuItem("退出");
        exitMenuItem.setMnemonic(
'E');
        
// 当“退出”时退出应用程序
        exitMenuItem.addActionListener(new ActionListener() {
            
public void actionPerformed(ActionEvent e) {
                exit();
            }

        }
);
        
        fileMenu.add(newMenuItem);
        fileMenu.add(openMenuItem);
        fileMenu.add(closeMenuItem);
        fileMenu.add(exitMenuItem);
        
        
//帮助菜单,就一个菜单项:关于
        JMenu helpMenu = new JMenu("帮助");
        fileMenu.setMnemonic(
'H');
        JMenuItem aboutMenuItem 
= new JMenuItem("关于");
        aboutMenuItem.setMnemonic(
'A');
        helpMenu.add(aboutMenuItem);
        
        JMenuBar menuBar 
= new JMenuBar();
        menuBar.add(fileMenu);
        menuBar.add(helpMenu);
        
        
// 设置菜单栏到主窗口
        this.setJMenuBar(menuBar);
    }

    
    
/**
     * 初始化工具栏
     
*/

    
private void initToolbar(){
        
// 后退按钮,退到前一页面。初始情况下该按钮不可用
        backButton = new JButton("后退");
        backButton.setEnabled(
false);
        backButton.addActionListener(
new ActionListener() {
            
public void actionPerformed(ActionEvent e) {
                back();
            }

        }
);
        
        
// 前进按钮,进到前一页面。初始情况下该按钮不可用
        forwardButton = new JButton("前进");
        forwardButton.setEnabled(
false);
        forwardButton.addActionListener(
new ActionListener() {
            
public void actionPerformed(ActionEvent e) {
                forward();
            }

        }
);
        
        
// 前进按钮,进到前一页面。初始情况下该按钮不可用
        JButton refreshButton = new JButton("刷新");
        refreshButton.addActionListener(
new ActionListener() {
            
public void actionPerformed(ActionEvent e) {
                reload();
            }

        }
);
        
        
// 主页按钮,打开主页
        JButton homeButton = new JButton("主页");
        homeButton.addActionListener(
new ActionListener() {
            
public void actionPerformed(ActionEvent e) {
                home();
            }

        }
);
        
        JToolBar toolbar 
= new JToolBar();
        toolbar.add(backButton);
        toolbar.add(forwardButton);
        toolbar.add(refreshButton);
        toolbar.add(homeButton);

        
// 输入网址的文本框
        urlField = new JTextField();
        
// 当用户输入网址、按下回车键时,触发该事件
        urlField.addActionListener(new ActionListener() {
                
public void actionPerformed(ActionEvent e) {
                    displayPage(urlField.getText());
                }

            }
);

        
// 地址标签
        toolbar.add(new JLabel("         地址:"));
        toolbar.add(urlField);

        
// 将工具栏放在主窗口的北部
        this.getContentPane().add(toolbar, BorderLayout.NORTH);
    }


    
/**
     * 设置浏览器是否在所有窗口都关闭时退出
     * 
@param b
     
*/

    
public static void setExitWhenLastWindowClosed(boolean b) {
        exitWhenLastWindowClosed 
= b;
    }


    
/**
     * 设置主页
     * 
@param home    新主页
     
*/

    
public void setHome(String home) {
        
this.home = home;
    }

    
/**
     * 获取主页
     
*/

    
public String getHome() {
        
return home;
    }


    
/**
     * 访问网址URL
     
*/

    
private boolean visit(URL url) {
        
try {
            String href 
= url.toString();
            
// 启动动画,当网页被加载完成时,触发propertyChanged事件,动画停止。
            startAnimation("加载 " + href + "");
            
            
// 设置显示HTML面板的page属性为待访问的URL,
            
// 在这个方法里,将打开URL,将URL的流显示在textPane中。
            
// 当完全打开后,该方法才结束。
            textPane.setPage(url); 
            
            
// 页面打开后,将浏览器窗口的标题设为URL
            this.setTitle(href);  
            
// 网址输入框的内容也设置为URL
            urlField.setText(href); 
            
return true;
        }
 catch (IOException ex) 
            
// 停止动画
            stopAnimation();
            
// 状态栏中显示错误信息
            messageLine.setText("不能打开页面:" + ex.getMessage());
            
return false;
        }

    }


    
/**
     * 浏览器打开URL指定的页面,如果成功,将URL放入历史列表中
     
*/

    
public void displayPage(URL url) {
        
// 尝试访问页面
        if (visit(url)) 
            
// 如果成功,则将URL放入历史列表中。
            history.add(url); 
            
int numentries = history.size();
            
if (numentries > MAX_HISTORY+10
                history 
= history.subList(numentries-MAX_HISTORY, numentries);
                numentries 
= MAX_HISTORY;
            }

            
// 将当前页面下标置为numentries-1
            currentHistoryPage = numentries - 1;
            
// 如果当前页面不是第一个页面,则可以后退,允许点击后退按钮。
            if (currentHistoryPage > 0){
                backButton.setEnabled(
true);
            }

        }

    }


    
/**
     * 浏览器打开字符串指定的页面
     * 
@param href    网址
     
*/

    
public void displayPage(String href) {
        
try {
            
// 默认为HTTP协议
            if (!href.startsWith("http://")){
                href 
= "http://" + href;
            }

            displayPage(
new URL(href));
        }

        
catch (MalformedURLException ex) {
            messageLine.setText(
"错误的网址: " + href);
        }

    }


    
/**
     * 打开本地文件
     
*/

    
public void openLocalPage() {
        
// 使用“懒创建”模式,当需要时,才创建文件选择器。
        if (fileChooser == null{
            fileChooser 
= new JFileChooser();
            
// 使用文件过滤器限制只能够HTML和HTM文件
            FileFilter filter = new FileFilter() {
                    
public boolean accept(File f) {
                        String fn 
= f.getName();
                        
if (fn.endsWith(".html"|| fn.endsWith(".htm")){
                            
return true;
                        }
else {
                            
return false;
                        }

                    }

                    
public String getDescription() 
                        
return "HTML Files"
                    }

                }
;
            fileChooser.setFileFilter(filter);
            
// 只允许选择HTML和HTM文件
            fileChooser.addChoosableFileFilter(filter);
        }


        
// 打开文件选择器
        int result = fileChooser.showOpenDialog(this);
        
// 如果确定打开文件,则在当前窗口中打开选择的文件
        if (result == JFileChooser.APPROVE_OPTION) {
            File selectedFile 
= fileChooser.getSelectedFile( );
            
try {
                displayPage(selectedFile.toURL());
            }
 catch (MalformedURLException e) {
                e.printStackTrace();
            }

        }

    }

    
/**
     * 后退,回到前一页
     
*/

    
public void back() {
        
if (currentHistoryPage > 0){
            
// 访问前一页
            visit((URL)history.get(--currentHistoryPage));
        }

        
// 如果当前页面下标大于0,允许后退
        backButton.setEnabled((currentHistoryPage > 0));
        
// 如果当前页面下标不是最后,允许前进
        forwardButton.setEnabled((currentHistoryPage < history.size()-1));
    }

    
/**
     * 前进,回到后一页
     
*/

    
public void forward() {
        
if (currentHistoryPage < history.size( )-1){
            visit((URL)history.get(
++currentHistoryPage));
        }

        
// 如果当前页面下标大于0,允许后退
        backButton.setEnabled((currentHistoryPage > 0));
        
// 如果当前页面下标不是最后,允许前进
        forwardButton.setEnabled((currentHistoryPage < history.size()-1));
    }

    
/**
     * 重新加载当前页面
     
*/

    
public void reload() {
        
if (currentHistoryPage != -1{
            
// 先显示为空白页
            textPane.setDocument(new javax.swing.text.html.HTMLDocument());
            
// 再访问当前页
            visit((URL)history.get(currentHistoryPage));
        }

    }

    
/**
     * 显示主页 
     
*/

    
public void home() {
        displayPage(getHome()); 
    }

    
/**
     * 打开新的浏览器窗口 
     
*/

    
public void newBrowser() {
        WebBrowser b 
= new WebBrowser();
        
// 新窗口大小和当前窗口一样大
        b.setSize(this.getWidth(), this.getHeight());
        b.setVisible(
true);
    }

    
/**
     * 关闭当前窗口,如果所有窗口都关闭,则根据exitWhenLastWindowClosed属性,
     * 判断是否退出应用程序
     
*/

    
public void close() {
        
// 先隐藏当前窗口,销毁窗口中的组件。
        this.setVisible(false);
        
this.dispose();
        
// 将当前打开窗口数减1。
        
// 如果所有窗口都已关闭,而且exitWhenLastWindowClosed为真,则退出
        
// 这里采用了synchronized关键字,保证线程安全
        synchronized(WebBrowser.class{    
            WebBrowser.numBrowserWindows
--
            
if ((numBrowserWindows==0&& exitWhenLastWindowClosed){
                System.exit(
0);
            }

        }

    }

    
/**
     * 退出应用程序
     
*/

    
public void exit() {
        
// 弹出对话框,请求确认,如果确认退出,则退出应用程序
        if ((JOptionPane.showConfirmDialog(this"你确定退出Web浏览器?""退出",
                JOptionPane.YES_NO_OPTION) 
== JOptionPane.YES_OPTION)){
            System.exit(
0);
        }

    }

    
/**
     * 实现HyperlinkListener接口。处理超连接事件
     
*/

    
public void hyperlinkUpdate(HyperlinkEvent e) {
        
// 获取事件类型
        HyperlinkEvent.EventType type = e.getEventType();
        
// 如果是点击了一个超连接,则显示被点击的连接
        if (type == HyperlinkEvent.EventType.ACTIVATED) {
            displayPage(e.getURL());
        }

        
// 如果是鼠标移动到超连接上,则在状态栏中显示超连接的网址
        else if (type == HyperlinkEvent.EventType.ENTERED) {
            messageLine.setText(e.getURL().toString());  
        }

        
// 如果是鼠标离开了超连接,则状态栏显示为空
        else if (type == HyperlinkEvent.EventType.EXITED) 
            messageLine.setText(
" ");
        }

    }


    
/**
     * 实现PropertyChangeListener接口。处理属性改变事件。
     * 显示HTML面板textPane的属性改变时,由该方法处理。
     * 当textPane调用完setPage方法时,page属性便改变了。
     
*/

    
public void propertyChange(PropertyChangeEvent e) {
        
if (e.getPropertyName().equals("page")) {
            
// 页面加载完毕时,textPane的page属性发生改变,此时停止动画。
            stopAnimation();
        }

    }


    
// 动画消息,显示在最底下状态栏标签上,用于反馈浏览器的状态
    String animationMessage;
    
// 动画当前的帧的索引
    int animationFrame = 0;
    
// 动画所用到的帧,是一些字符。
    String[] animationFrames = new String[] {
        
"-""\\""|""/""-""\\""|""/"
        
","".""o""0""O""#""*""+"
    }
;

    
/**
     * 新建一个Swing的定时器,每个125毫秒更新一次状态栏标签的文本
     
*/

    javax.swing.Timer animator 
=
        
new javax.swing.Timer(125new ActionListener() {
                
public void actionPerformed(ActionEvent e) {
                    animate(); 
                }

            }
);

    
/**
     * 显示动画的当前帧到状态栏标签上,并将帧索引后移
     
*/

    
private void animate() {
        String frame 
= animationFrames[animationFrame++];
        messageLine.setText(animationMessage 
+ " " + frame);
        animationFrame 
= animationFrame % animationFrames.length;
    }


    
/**
     * 启动动画
     
*/

    
private void startAnimation(String msg) {
        animationMessage 
= msg;
        animationFrame 
= 0
        
// 启动定时器
        animator.start();
    }


    
/**
     * 停止动画
     
*/

    
private void stopAnimation() {  
        
// 停止定时器
        animator.stop();
        messageLine.setText(
" ");
    }

    
    
public static void main(String[] args) throws IOException {
        
// 设置浏览器,当所有浏览器窗口都被关闭时,退出应用程序
        WebBrowser.setExitWhenLastWindowClosed(true);
        
// 创建一个浏览器窗口
        WebBrowser browser = new WebBrowser(); 
        
// 设置浏览器窗口的默认大小
        browser.setSize(800600);
        
// 显示窗口
        browser.setVisible(true); 

        
// 打开主页
        browser.displayPage(browser.getHome());
    }

}



                                                                                                       --    学海无涯