Javafx实现浏览器

        浏览器是一种计算机程序,主要用于显示互联网上的网页。通过浏览器,用户可以访问各种网站、搜索引擎、在线应用程序、社交媒体等。常见的浏览器包括Google Chrome、Mozilla Firefox、Safari、Microsoft Edge、Opera等。浏览器的功能不仅限于浏览网页,还包括下载文件、管理书签、保存密码、清除浏览数据等。浏览器已成为人们日常生活中必不可少的工具之一。

        下面我们使用JavaFx来实现一下浏览器,代码如下:


import javafx.application.Application;
import javafx.concurrent.Worker;
import javafx.scene.Scene;
import javafx.scene.control.Label;
import javafx.scene.control.TextField;
import javafx.scene.image.Image;
import javafx.scene.input.KeyCode;
import javafx.scene.layout.*;
import javafx.scene.web.WebEngine;
import javafx.scene.web.WebHistory;
import javafx.scene.web.WebView;
import javafx.stage.Stage;
import lombok.extern.slf4j.Slf4j;

@Slf4j
public class BrowserApplication extends Application {
    private static String URL = "https://www.fmill.cn/";
    private String url = URL;
    private String content;

    private Stage stage;
    private WebEngine engine;
    private TextField urlField;

    public BrowserApplication() {
    }

    public BrowserApplication(String url) {
        this.url = url;
        URL = url;
    }

    public BrowserApplication setAutoRefresh() {
        return this;
    }

    @Override
    public void start(Stage primaryStage) {
        stage = primaryStage;
        AnchorPane root = new AnchorPane();
//
        HBox topBox = new HBox();
        topBox.setPrefHeight(30);
        topBox.setPrefWidth(1000);
//        topBox.setStyle("-fx-border-color: #999");

        int len = 60;
        urlField = new TextField();
        urlField.setPrefWidth(916 - len);
        urlField.setPrefHeight(30);
        setSearchTextFieldStyle(urlField);

        Image iconImage = JavaFxUtils.getIconImage();
        Label homeLabel = newLabel(JavaFxUtils.getHomeImage());
        Label backLabel = newLabel(JavaFxUtils.getPrevImage());
        Label nextLabel = newLabel(JavaFxUtils.getNextImage());
        Label freshLabel = newLabel(JavaFxUtils.getFreshImage());
        Label moreLabel = newLabel(JavaFxUtils.getMoreImage());

        topBox.getChildren().addAll(backLabel, nextLabel, freshLabel, homeLabel, urlField, moreLabel);

        WebView browser = new WebView();
        browser.setPrefWidth(1000);
        browser.setPrefHeight(800);
        AnchorPane.setTopAnchor(browser, 30.0);
        //获取web浏览器引擎内核对象
        engine = browser.getEngine();

        engine.getLoadWorker().stateProperty().addListener((obs, oldValue, newValue) -> {
            System.out.println(oldValue + "->" + newValue);
            reload();
            if (newValue == Worker.State.SUCCEEDED) {
                System.out.println("finished loading");

            }
        });

        urlField.setText(this.url);
        urlField.setOnAction(e -> {
            href(urlField.getText());
        });
//        urlField.textProperty().bind(engine.locationProperty());

        System.out.println("是否开启脚本:" + engine.isJavaScriptEnabled());
        engine.userAgentProperty().addListener(((observable, oldValue, newValue) -> {
            System.out.println("userAgent:" + oldValue + "," + newValue);
        }));
        if (content != null) {
            engine.loadContent(content);
        } else {
            href(url);
        }
        engine.locationProperty().addListener((event, o, n) -> {
            System.out.println(o + "->" + n);
        });

        root.getChildren().addAll(topBox, browser);

        primaryStage.setTitle("小筱浏览器");
        stage.getIcons().add(iconImage);
        Scene scene = new Scene(root);
        primaryStage.setScene(scene);
        primaryStage.setWidth(1020);
        primaryStage.setHeight(850);
        primaryStage.show();
        primaryStage.widthProperty().addListener((event, o, n) -> {
            if (n != null) {
                browser.setPrefWidth(n.doubleValue() - 20);
                topBox.setPrefWidth(n.doubleValue() - 10);
                urlField.setPrefWidth(n.doubleValue() - 110 - len);
            }
        });
        primaryStage.heightProperty().addListener((event, o, n) -> {
            browser.setPrefHeight(n.doubleValue() - 50);
        });
        homeLabel.setOnMouseClicked(event -> {
            System.out.println("首页");
            href("https://www.fmill.cn");
        });
        backLabel.setOnMouseClicked(event -> {
            WebHistory history = engine.getHistory();
            if (history.getCurrentIndex() > 0) {
                history.go(-1);
            }
        });
        nextLabel.setOnMouseClicked(event -> {
            WebHistory history = engine.getHistory();
            if (history.getCurrentIndex() < history.getEntries().size() - 1) {
                history.go(+1);
            }
        });
        freshLabel.setOnMouseClicked(event -> {
            String location = engine.getLocation();
            href(location);
        });

        root.setOnKeyPressed(event -> {
            if (event.getCode().equals(KeyCode.F5)) {
                System.out.println("刷新页面");
                engine.reload();
                String location = engine.getLocation();
                System.out.println(location);
                this.url = location;
                URL = this.url;
            }
        });
        engine.onStatusChangedProperty().addListener((event, o, n) -> {
            System.out.println("onStatusChangedProperty");
        });
        primaryStage.setOnCloseRequest(event -> {
            primaryStage.close();
        });
    }

    private void href(String openUrl) {
        engine.load(openUrl);
        this.url = openUrl;
        URL = this.url;
        this.urlField.setText(openUrl);
    }

    private void reload() {
        String location = engine.getLocation();
        WebHistory history = engine.getHistory();
        System.out.println(history.getCurrentIndex());
        if (!this.url.equals(location)) {
            System.out.println("网址发生改变...");
            this.url = location;
            URL = this.url;
            this.urlField.setText(location);
        } else {
            System.out.println("网址未发生变动");
        }
    }

    private static Label newLabel(Image image) {
        Label label = new Label();
        label.setBackground(getBackground(image));
        label.setPrefWidth(30);
        label.setPrefHeight(30);
        label.setMinWidth(30);
        label.setMinHeight(30);
        return label;
    }

    public static Background getBackground(Image image) {
        return new Background(new BackgroundImage(image, BackgroundRepeat.NO_REPEAT, BackgroundRepeat.NO_REPEAT, BackgroundPosition.CENTER, BackgroundSize.DEFAULT));
    }

    private void setSearchTextFieldStyle(TextField urlField) {
        urlField.setStyle("-fx-border-color: #e2e2e2;-fx-background-color: #e2e2e2;-fx-border-radius: 10px;-fx-background-radius: 10px;-fx-highlight-fill: #a4adc7;-fx-highlight-text-fill: #000000;");
        urlField.setOnMousePressed(event ->
                urlField.setStyle("-fx-border-color: #29aff8;-fx-background-color: #fff;-fx-border-radius: 10px;-fx-highlight-fill: #a4adc7;-fx-highlight-text-fill: #000000;"));
        urlField.setOnMouseExited(event ->
                urlField.setStyle("-fx-border-color: #ced0d5;-fx-background-color: #fff;-fx-border-radius: 10px;-fx-highlight-fill: #a4adc7;-fx-highlight-text-fill: #000000;"));
    }

    public void setUrl(String url) {
        this.url = url;
        URL = this.url;
    }

    public BrowserApplication setContent(String content) {
        this.content = content;
        return this;
    }
}

        其中获取Image这里需要自己定义:

        图标在下面,有需要的话自己抓取:

        输入百度链接,效果如下:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/220658.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

无线网卡填坑记

没想到我安装无线网卡这么波澜起伏~ 起因 近来刚在电脑上玩完了 Dishonored 2&#xff0c;紧接着继续着我的刺客信条之旅。总是觉得键盘鼠标玩起来不爽&#xff0c;还是手柄玩这种游戏才舒服。突然&#xff0c;灵光一现&#xff0c;我想到正好有闲置的 Switch 掌机没怎么玩&am…

【代码随想录】算法训练计划39

dp 1、62. 不同路径 题目&#xff1a; 求路径方案多少个 思路&#xff1a; 这道题就有点dp了哈 func uniquePaths(m int, n int) int {//dp&#xff0c;写过,代表的是多少种// 初始化dp : make([][]int, m)for i : range dp {dp[i] make([]int, n)dp[i][0] 1 // 代表到…

【数据结构】图<简单认识图>

对于下面的内容&#xff0c;大家着重观察和理解图即可&#xff0c;可以直接绕过一些文字性的概念&#xff0c;对图有一个大概的认识。 图 简单认识图图的定义有向图和无向图完全图无向完全图有向完全图 图的基本存储结构邻接矩阵存储邻接矩阵的优点 网络的邻接矩阵邻接表无向图…

看懂lscpu的输出

文章目录 1. lscpu1.1 Architecture1.2 逻辑核心数1.3 缓存1.4 CPU型号1.5 NUMA架构1.5.1 CPU多核架构1.5.2 多CPU Socket架构 2. cat /proc/cpuinfo2.1 关键字段 1. lscpu 通过lscpu查看当前系统的CPU信息。 [hadoopserver3 ~]$ lscpuArchitecture: x86_64 …

混音编曲软件tudio One 6.5.1 保姆级安装教程

根据软件大数据显示De-Esser驯服人声嘶嘶声和其他高频声音&#xff0c;和其他 Studio One 中新的去实体插件一样高效且直观易用&#xff0c;使用“收听”按钮查找有问题的频率&#xff0c;然后使用相关的旋钮和 S-Mon 功能拨入 S-Reduce 量即可。实际上我们可以这样讲工作流和协…

Linux(15):SELinux 初探

什么是 SELinux SELinux 是【Security Enhanced Linux】的缩写&#xff0c;字面上的意义就是安全强化的 Linux。 SELinux 是由美国国家安全局(NSA)开发的&#xff0c;开发原因&#xff1a;因为很多企业界发现&#xff0c;通常系统出现问题的原因大部分都在于【内部员工的资源…

Redis的三种消息队列实现方式

目录 前言 List实现消息队列 PubSub消息队列 Stream消息队列 三种实现方式对比 前言 为什么要使用Redis的消息队列&#xff1f; 成本低&#xff0c;对于RabbitMQ或是Kafka来说&#xff0c;已经是重量级的消息队列。 Redis的三种实现方式&#xff1a; List结构&#xff1…

VSC改造MD编辑器及图床方案分享

VSC改造MD编辑器及图床方案分享 用了那么多md编辑器&#xff0c;到头来还是觉得VSC最好用。这次就来分享一下我的blog文件编辑流吧。 这篇文章包括&#xff1a;VSC下md功能扩展插件推荐、图床方案、blog文章管理方案 VSC插件 Markdown All in One Markdown Image - 粘粘图片…

在python的Scikit-learn库中,可以使用train_test_split函数来划分训练集和测试集。

文章目录 一、在Scikit-learn库中&#xff0c;可以使用train_test_split函数来划分训练集和测试集总结 一、在Scikit-learn库中&#xff0c;可以使用train_test_split函数来划分训练集和测试集 在Scikit-learn库中&#xff0c;可以使用train_test_split函数来划分训练集和测试…

【网络安全】红蓝对抗之企业互联网安全防护

01 什么是“红蓝对抗”&#xff1f; “红蓝对抗”最早起源于古罗马军队&#xff0c;在沙盘中用红色和蓝色来代表敌人和自己&#xff0c;他们认为蓝色代表勇敢和忠诚&#xff0c;红色代表血腥和暴力&#xff0c;所以选择用蓝色代表自己。 在中国&#xff0c;由于传统习俗与文化…

一、技术体系结构

本章概要 总体技术体系框架概念和理解 1.1 总体技术体系 单一架构一个项目&#xff0c;一个工程&#xff0c;导出为一个war包&#xff0c;在一个Tomcat上运行。也叫all in one。 单一架构&#xff0c;项目主要应用技术框架为&#xff1a;Spring , SpringMVC , Mybatis 分布…

Python如何传递任意数量的实参及什么是返回值

Python如何传递任意数量的实参 传递任意数量的实参 形参前加一个 * &#xff0c;Python会创建一个已形参为名的空元组&#xff0c;将所有收到的值都放到这个元组中&#xff1a; def make_pizza(*toppings):print("\nMaking a pizza with the following toppings: "…

【ArcGIS Pro】探索性插值无法覆盖所需shp范围

做个小记录自用&#xff0c;实际不准。 1 看看就行 pro插值 看看过程就行。有详细过程&#xff0c;类似tutorial https://learn.arcgis.com/zh-cn/projects/interpolate-temperatures-using-the-geostatistical-wizard/ 2 注意用投影坐标系 wgs84转投影坐标系 https://blog…

SR锁存器—>带EN的SR锁存器—>D锁存器—>边沿触发式D触发器—>寄存器

其实选择与非门当做构成SR锁存器的基本逻辑电路是有漏洞的&#xff0c;所以才导致了后续的都为低电平的时候&#xff0c;Q和非Q都是亮起的。但是我们设计的初衷是&#xff1a;Q和非Q是互斥的&#xff0c;是不能同时亮起的&#xff0c;且为了达到这一点&#xff0c;要使得其中两…

用友NC JiuQiClientReqDispatch反序列化RCE漏洞复现

0x01 产品简介 用友NC是一款企业级ERP软件。作为一种信息化管理工具,用友NC提供了一系列业务管理模块,包括财务会计、采购管理、销售管理、物料管理、生产计划和人力资源管理等,帮助企业实现数字化转型和高效管理。 0x02 漏洞概述 用友 NC JiuQiClientReqDispatch 接口存在…

EasyRecovery14破解版 v14.0.0.4 官方免费版(含激活码)

软件介绍 EasyRecovery14高级版是一款功能强大的数据恢复软件&#xff0c;软件对比家庭版本它的使用更加广泛&#xff0c;在恢复数据方面软件可以做到最完整的损失恢复&#xff0c;无论是文档、音乐、软件都可以一键恢复&#xff0c;同时软件还可以对文件的名字、后缀进行修改…

龙芯loongarch64服务器编译安装tokenizers

1、简介 Hugging Face 的 Tokenizers 库提供了一种快速和高效的方式来处理(即分词)自然语言文本,用于后续的机器学习模型训练和推理。这个库提供了各种各样的预训练分词器,如 BPE、Byte-Pair Encoding (Byte-Level BPE)、WordPiece 等,这些都是现代 NLP 模型(如 BERT、GP…

浅谈ArrayBuffer、Blob和File、FileReader

ArrayBuffer、Blob和File都是JavaScript中处理二进制数据的对象。 ArrayBuffer 用于表示一个通用的、固定长度的原始二进制数据缓冲区。它不能直接操作缓冲区中的数据&#xff0c;而需要通过一个类型化数组TypedArray&#xff08;如Int8Array、Uint8Array等&#xff09;或者一…

你好!哈希表【JAVA】

1.初识&#x1f3b6;&#x1f3b6;&#x1f3b6; 它基本上是由一个数组和一个哈希函数组成的。哈希函数将每个键映射到数组的特定索引位置&#xff0c;这个位置被称为哈希码。当我们需要查找一个键时&#xff0c;哈希函数会计算其哈希码并立即返回结果&#xff0c;因此我们可以…

消息中间件之间的区别

一.单机吞吐量 ActiveMQ&#xff1a;万级&#xff0c;吞吐量比RocketMQ和Kafka要低了一个数量级 RabbitMQ&#xff1a;万级&#xff0c;吞吐量比RocketMQ和Kafka要低了一个数量级 RocketMQ&#xff1a;10万级&#xff0c;RocketMQ也是可以支撑高吞吐的一种MQ Kafka&#xff…