pyqt自制简单浏览器(python)

确保已安装 PyQt5 和 PyQtWebEngine 库。

import sys
from PyQt5.QtCore import QUrl
from PyQt5.QtWidgets import QApplication, QMainWindow, QToolBar, QLineEdit, QAction, QListWidget, QVBoxLayout, QDialog, QMessageBox, QInputDialog, QTabWidget
from PyQt5.QtWebEngineWidgets import QWebEngineView, QWebEnginePage
import json
import os

class WebEnginePage(QWebEnginePage):
    def __init__(self, parent=None, browser_window=None):
        super().__init__(parent)
        self.browser_window = browser_window  # 保存 WebBrowser 实例

    def createWindow(self, type):
        """重写 createWindow 方法以支持在新标签页中打开链接"""
        if self.browser_window:
            new_tab = self.browser_window.add_tab(QUrl("about:blank"), "新标签页")
            return new_tab.page()
        return None

class WebBrowser(QMainWindow):
    def __init__(self):
        super().__init__()
        self.setWindowTitle("简单浏览器")
        self.setGeometry(100, 100, 1000, 800)

        # 初始化数据
        self.history = []
        self.bookmarks = []
        self.homepage = "https://www.google.com"  # 默认主页
        self.load_data()

        # 创建标签页
        self.tabs = QTabWidget()
        self.tabs.setTabsClosable(True)
        self.tabs.tabCloseRequested.connect(self.close_tab)
        self.setCentralWidget(self.tabs)

        # 创建初始标签页
        self.add_tab(QUrl(self.homepage), "主页")

        # 创建导航栏
        self.create_navbar()

    def create_navbar(self):
        """创建导航栏"""
        navbar = QToolBar("导航栏")
        self.addToolBar(navbar)

        # 主页按钮
        home_btn = QAction("🏠", self)
        home_btn.triggered.connect(self.go_home)
        navbar.addAction(home_btn)

        # 返回按钮
        back_btn = QAction("←", self)
        back_btn.triggered.connect(self.go_back)
        navbar.addAction(back_btn)

        # 前进按钮
        forward_btn = QAction("→", self)
        forward_btn.triggered.connect(self.go_forward)
        navbar.addAction(forward_btn)

        # 刷新按钮
        refresh_btn = QAction("↻", self)
        refresh_btn.triggered.connect(self.refresh)
        navbar.addAction(refresh_btn)

        # URL 输入框
        self.url_entry = QLineEdit()
        self.url_entry.returnPressed.connect(self.load_url)
        navbar.addWidget(self.url_entry)

        # 访问按钮
        go_btn = QAction("访问", self)
        go_btn.triggered.connect(self.load_url)
        navbar.addAction(go_btn)

        # 收藏按钮
        bookmark_btn = QAction("⭐", self)
        bookmark_btn.triggered.connect(self.add_bookmark)
        navbar.addAction(bookmark_btn)

        # 历史记录按钮
        history_btn = QAction("历史", self)
        history_btn.triggered.connect(self.show_history)
        navbar.addAction(history_btn)

        # 收藏夹按钮
        bookmarks_btn = QAction("收藏夹", self)
        bookmarks_btn.triggered.connect(self.show_bookmarks)
        navbar.addAction(bookmarks_btn)

        # 设置主页按钮
        set_home_btn = QAction("设置主页", self)
        set_home_btn.triggered.connect(self.set_homepage)
        navbar.addAction(set_home_btn)

        # 新标签页按钮
        new_tab_btn = QAction("➕", self)
        new_tab_btn.triggered.connect(lambda: self.add_tab(QUrl(self.homepage), "新标签页"))
        navbar.addAction(new_tab_btn)

    def add_tab(self, url, title):
        """添加新标签页"""
        browser = QWebEngineView()
        browser.setUrl(url)
        # 使用自定义的 WebEnginePage 处理链接点击事件,并传递 WebBrowser 实例
        browser.setPage(WebEnginePage(browser, self))
        index = self.tabs.addTab(browser, title)
        self.tabs.setCurrentIndex(index)

        # 监听网页标题变化,更新标签页标题
        browser.titleChanged.connect(lambda title, idx=index: self.tabs.setTabText(idx, title))
        return browser

    def close_tab(self, index):
        """关闭标签页"""
        if self.tabs.count() > 1:
            self.tabs.removeTab(index)

    def get_current_browser(self):
        """获取当前标签页的浏览器实例"""
        return self.tabs.currentWidget()

    def load_url(self):
        """加载 URL"""
        url = self.url_entry.text().strip()
        if not url.startswith(("http://", "https://")):
            url = "https://" + url
        self.get_current_browser().setUrl(QUrl(url))
        self.add_to_history(url)

    def go_home(self):
        """返回主页"""
        self.get_current_browser().setUrl(QUrl(self.homepage))

    def set_homepage(self):
        """设置主页"""
        url, ok = QInputDialog.getText(self, "设置主页", "请输入主页 URL:", text=self.homepage)
        if ok and url:
            if not url.startswith(("http://", "https://")):
                url = "https://" + url
            self.homepage = url
            self.save_data()

    def go_back(self):
        """返回上一页"""
        self.get_current_browser().back()

    def go_forward(self):
        """前进到下一页"""
        self.get_current_browser().forward()

    def refresh(self):
        """刷新页面"""
        self.get_current_browser().reload()

    def add_to_history(self, url):
        """添加到历史记录"""
        self.history.append(url)
        self.save_data()

    def add_bookmark(self):
        """添加收藏"""
        url = self.url_entry.text().strip()
        if url and url not in self.bookmarks:
            self.bookmarks.append(url)
            self.save_data()
            QMessageBox.information(self, "成功", "已添加到收藏夹")

    def show_history(self):
        """显示历史记录"""
        history_window = QDialog(self)
        history_window.setWindowTitle("历史记录")
        history_window.setGeometry(200, 200, 400, 300)

        history_list = QListWidget()
        for url in self.history:
            history_list.addItem(url)

        layout = QVBoxLayout()
        layout.addWidget(history_list)
        history_window.setLayout(layout)

        history_window.exec_()

    def show_bookmarks(self):
        """显示收藏夹"""
        bookmarks_window = QDialog(self)
        bookmarks_window.setWindowTitle("收藏夹")
        bookmarks_window.setGeometry(200, 200, 400, 300)

        bookmarks_list = QListWidget()
        for url in self.bookmarks:
            bookmarks_list.addItem(url)

        layout = QVBoxLayout()
        layout.addWidget(bookmarks_list)
        bookmarks_window.setLayout(layout)

        bookmarks_window.exec_()

    def load_data(self):
        """加载历史记录和收藏夹数据"""
        if os.path.exists("browser_data.json"):
            with open("browser_data.json", "r") as f:
                data = json.load(f)
                self.history = data.get("history", [])
                self.bookmarks = data.get("bookmarks", [])
                self.homepage = data.get("homepage", "https://www.google.com")

    def save_data(self):
        """保存历史记录和收藏夹数据"""
        data = {
            "history": self.history,
            "bookmarks": self.bookmarks,
            "homepage": self.homepage
        }
        with open("browser_data.json", "w") as f:
            json.dump(data, f)

if __name__ == "__main__":
    app = QApplication(sys.argv)
    browser = WebBrowser()
    browser.show()
    sys.exit(app.exec_()) 

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

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

相关文章

【人工智能】如何选择合适的大语言模型,是能否提高工作效率的关键!!!

DeepSeek R1入门指南 导读一、提示语差异1.1 指令侧重点不同1.2 语言风格差异1.3 知识运用引导不同 二、挑选原则2.1 模型选择2.2 提示语设计2.3 避免误区 结语 导读 大家好,很高兴又和大家见面啦!!! 在前面的内容中&#xff0c…

​矩阵元素的“鞍点”​

题意: 一个矩阵元素的“鞍点”是指该位置上的元素值在该行上最大、在该列上最小。 本题要求编写程序,求一个给定的n阶方阵的鞍点。 输入格式: 输入第一行给出一个正整数n(1≤n≤6)。随后n行,每行给出n个整数…

ChatGPT搜索免费开放:AI搜索引擎挑战谷歌霸主地位全面分析

引言 2025年2月6日,OpenAI宣布ChatGPT搜索功能向所有用户免费开放,且无需注册登录。这一重大举措在搜索引擎行业引发巨大反响,有观点认为"谷歌搜索时代即将结束"。本文将深入分析ChatGPT生成式AI搜索对谷歌搜索业务及全球搜索市场…

NO.18十六届蓝桥杯备战|循环嵌套|乘法表|斐波那契|质数|水仙花数|(C++)

循环嵌套 循环嵌套的使⽤ while , do while , for ,这三种循环往往会嵌套在⼀起才能更好的解决问题,就是我们所说的:循环嵌套。这三种循环都可以任意嵌套使⽤ ⽐如: 写⼀个代码,打印⼀个乘法⼝…

国际互联网安全日|Web3 世界的安全挑战与防护指南

2025 年 2 月 11 日是全球 “国际互联网安全日”(Safer Internet Day)。当我们跨越 Web2 迈入 Web3 时代,互联网安全的内涵也在悄然改变。在 Web2 时代,我们主要关注社交媒体隐私泄露、账号密码被盗、网络诈骗等传统安全问题。而在…

DeepseeK自动写作,自动将回答导出文档

在使用 DeepseeK 进行内容生成时,您是否也遇到了答案导出的困扰?无论是内容创作、数据分析还是项目报告,快速、高效地将生成的答案导出是提升工作效率的关键。本文将为您提供简单易行的解决方案,助您轻松实现 DeepseeK 答案的导出…

deep seek

1.介绍:DeepSeek是一款由国内人工智能公司研发的大型语言模型,拥有强大的自然语言处理能力,能够理解并回答问题,还能辅助写代码、整理资料和解决复杂的数学问题。免费开源,媲美ChatGPT 最近最火爆的AI对话程序。 www.deepseek.com…

数据结构中的邻接矩阵

一、概念 邻接矩阵(Adjacency Matrix)是图(Graph)的一种表示方法,用于描述图中顶点之间的连接关系。它是一种常见的数据结构,特别适用于表示稠密图(即边数接近于顶点数平方的图)。 …

微软AutoGen高级功能——Selector Group Chat

介绍 大家好,这次给大家分享的内容是微软AutoGen框架的高级功能Selector Group Chat(选择器群聊),"选择器群聊"我在给大家分享的这篇博文的代码中有所体现微软AutoGen介绍——Custom Agents创建自己的Agents-CSDN博客,但是并没有详…

web前端开发中vscode常用的快捷键

1.快速复制一行 快捷键: shiftalt 下箭头(上箭头) 或者 ctrlc 然后 ctrlv 2.选定多个相同的单词 快捷键: ctrl d 先双击选定一个单词,然后按下 ctrl d 可以往下依次选择相同的单词。 这样同时修改相同的单词 3.全局替换某单词 当我们一个…

网络安全要学python 、爬虫吗

网络安全其实并不复杂,只是比普通开发岗位要学习的内容多一点。无论是有过编程基础还是零基础的都可以学习的。网络安全目前可就业的岗位从技术上可分为两部分:web安全和二进制逆向安全。web安全是网络安全的入门方向,内容简单,就…

深入解析哈希表:原理、实现与应用

目录 一、哈希表是什么? 1.1 基本概念 1.2 哈希表的工作原理 二、哈希表的使用方法 2.1 C 标准库中的哈希表 示例:std::unordered_map 的基本用法 2.2 自定义哈希函数 示例:自定义哈希函数 三、什么时候使用哈希表? 3.1…

CentOS-Stream 9更换RT实时内核

文章目录 1 安装环境1.1 Centos9原生内核示意 2 下载实时内核3 CentOS更换阿里YUM源3.1 更换源3.2 添加文件内容3.2.1 centos.repo3.2.2 centos-addons.repo 3.3 yum源生效 4 安装epel-release5 安装必要库和软件6 配置内核6.1 更改内核文件权限6.2 使用tar命令解压内核源码文件…

微软AutoGen高级功能——Magentic-One

介绍 大家好,博主又来给大家分享知识了,这次给大家分享的内容是微软AutoGen框架的高级功能Magentic-One。那么它是用来做什么的或它又是什么功能呢,我们直接进入正题。 Magentic-One Magnetic-One是一个通用型多智能体系统,用于…

国产编辑器EverEdit - 极简追梦人的福音:迷你查找

1 迷你查找 1.1 应用场景 某些场景下,用户不希望调出复杂的查找对话框,此时可以使用迷你查找窗口。 1.2 使用方法 选择主菜单查找 -> 迷你查找,或使用快捷键Ctrl Alt F,会在右上角弹出迷你查找窗口,如下图所示…

【第4章:循环神经网络(RNN)与长短时记忆网络(LSTM)— 4.5 序列标注与命名实体识别】

一、引言 嘿,各位技术小伙伴们!今天咱们要来深入聊聊循环神经网络(RNN)和长短时记忆网络(LSTM),这俩在序列标注和命名实体识别领域那可是相当厉害的角色。咱就从最基础的概念开始,一步步揭开它们神秘的面纱,看看它们到底是怎么在实际应用中发挥巨大作用的。 二、序列…

AIGC与AICG的区别解析

目录 一、AIGC(人工智能生成内容) (一)定义与内涵 (二)核心技术与应用场景 (三)优势与挑战 二、AICG(计算机图形学中的人工智能) (一&#x…

DeepSeek 助力 Vue 开发:打造丝滑的卡片(Card)

前言:哈喽,大家好,今天给大家分享一篇文章!并提供具体代码帮助大家深入理解,彻底掌握!创作不易,如果能帮助到大家或者给大家一些灵感和启发,欢迎收藏关注哦 💕 目录 Deep…

采购订单不计算MRP的供给问题

采购订单不计算MRP的供给问题。MD04的例外信息20 20的例外信息的意思:重复需求,建议取消。也就是不算MRP的供给。 MRP最大运行期间的配置 MRP最大运行期间是指MRP运行时考虑的时间范围。配置路径为: 事务码:OMDX 或通过路径&…

DeepSeek处理自有业务的案例:让AI给你写一份小众编辑器(EverEdit)的语法着色文件

1 DeepSeek处理自有业务的案例:让AI给你写一份小众编辑器(EverEdit)的语法着色文件 1.1 背景 AI能力再强,如果不能在企业的自有业务上产生助益,那基本也是一无是处。将企业的自有业务上传到线上训练,那是脑子进水的做法&#xff…