探索文件与交互:使用PyQt5构建一个高级文件选择器

news2025/1/23 4:55:47

在当今的应用程序开发中,文件管理和交互是一个重要的组成部分。特别是对于桌面应用程序,提供一个直观、功能丰富的文件选择器是提高用户体验的关键。

本篇博客,我将介绍如何使用Python和PyQt5来构建一个高级的文件选择器,它不仅能浏览文件,还能预览图片,编辑文本文件,并提供基本的右键菜单操作。

关键功能

  • 文件浏览:使用QColumnViewQFileSystemModel展示文件系统。
  • 图片预览:选中图片文件时,能在界面中预览。
  • 文本编辑:选中文本文件时,能在界面中进行编辑。
  • 保存编辑内容:编辑文本文件后,提供保存功能。
  • 右键菜单:提供自定义的右键菜单,实现文件的打开和查看所在文件夹。

设计思路

使用PyQt5的强大功能,我们可以轻松创建出复杂的用户界面。首先,我们使用QColumnView来展示文件系统的层级结构,它能提供直观的列式浏览体验。接着,通过QFileSystemModel来管理和展示文件系统中的数据。

图片预览和文本编辑功能是通过判断文件类型来实现的。如果选中的是图片文件(如jpg、png等),程序会在一个QLabel中显示该图片。如果选中的是文本文件(如txt、py等),则可以在QTextEdit中编辑,并通过一个保存按钮将更改保存回文件。

右键菜单是通过setContextMenuPolicycustomContextMenuRequested信号实现的。当用户在文件上点击右键时,会弹出一个自定义菜单,提供打开文件或文件所在文件夹的选项。

代码实现

以下是完整的代码实现: 

import sys
import os
from PyQt5.QtWidgets import QApplication, QMainWindow, QColumnView, QFileSystemModel, QLabel, QTextEdit, QPushButton, \
    QVBoxLayout,QHBoxLayout, QWidget, QMenu
from PyQt5.QtGui import QPixmap, QDesktopServices
from PyQt5.QtCore import QDir, QFileInfo, QUrl, Qt
from PyQt5.Qt import QSplitter


class FileExplorer(QMainWindow):
    def __init__(self):
        super().__init__()
        self.setWindowTitle("文件选择器")
        self.setGeometry(100, 100, 1000, 600)

        self.model = QFileSystemModel()
        self.model.setRootPath(QDir.rootPath())

        self.columnView = QColumnView()
        self.columnView.setModel(self.model)
        self.columnView.clicked.connect(self.on_file_selected)

        self.columnView.setContextMenuPolicy(Qt.CustomContextMenu)
        self.columnView.customContextMenuRequested.connect(self.open_context_menu)

        self.imageLabel = QLabel("图片预览")
        self.imageLabel.setScaledContents(True)
        self.textEdit = QTextEdit()
        self.saveButton = QPushButton("保存")
        self.saveButton.clicked.connect(self.save_file)

        rightLayout = QVBoxLayout()
        rightLayout.addWidget(self.imageLabel)
        rightLayout.addWidget(self.textEdit)
        rightLayout.addWidget(self.saveButton)

        self.rightWidget = QWidget()
        self.rightWidget.setLayout(rightLayout)

        splitter = QSplitter(Qt.Vertical)
        splitter.addWidget(self.columnView)
        splitter.addWidget(self.rightWidget)
        splitter.setStretchFactor(1, 1)

        centralWidget = QWidget()
        centralWidget.setLayout(QVBoxLayout())
        centralWidget.layout().addWidget(splitter)
        self.setCentralWidget(centralWidget)

    def on_file_selected(self, index):
        path = self.model.filePath(index)
        fileInfo = QFileInfo(path)

        if fileInfo.isFile():
            if fileInfo.suffix().lower() in ['png', 'jpg', 'jpeg', 'bmp', 'gif']:
                self.show_image_preview(path)
            elif fileInfo.suffix().lower() in ['txt', 'py', 'html', 'css', 'js', 'cs']:
                self.show_text_editor(path)
            else:
                self.clear_previews()

    def show_image_preview(self, path):
        self.textEdit.hide()
        self.saveButton.hide()
        self.imageLabel.setPixmap(QPixmap(path))
        self.imageLabel.show()

    def show_text_editor(self, path):
        self.imageLabel.hide()
        self.textEdit.setPlainText("")
        self.textEdit.show()
        self.saveButton.show()
        with open(path, 'r', encoding="utf-8") as file:
            self.textEdit.setText(file.read())
        self.currentTextFilePath = path

    def save_file(self):
        with open(self.currentTextFilePath, 'w', encoding='utf-8') as file:
            file.write(self.textEdit.toPlainText())

    def clear_previews(self):
        self.imageLabel.clear()
        self.textEdit.clear()
        self.imageLabel.hide()
        self.textEdit.hide()
        self.saveButton.hide()

    def open_context_menu(self, position):
        index = self.columnView.indexAt(position)
        if not index.isValid():
            return

        menu = QMenu()
        openAction = menu.addAction("打开")
        openFolderAction = menu.addAction("打开所在文件夹")

        action = menu.exec_(self.columnView.mapToGlobal(position))

        if action == openAction:
            self.open_file(index)
        elif action == openFolderAction:
            self.open_file_folder(index)

    def open_file(self, index):
        path = self.model.filePath(index)
        QDesktopServices.openUrl(QUrl.fromLocalFile(path))

    def open_file_folder(self, index):
        path = self.model.filePath(index)
        QDesktopServices.openUrl(QUrl.fromLocalFile(os.path.dirname(path)))


if __name__ == '__main__':
    app = QApplication(sys.argv)
    ex = FileExplorer()
    ex.show()
    sys.exit(app.exec_())

结语

这个文件选择器是一个展示PyQt5框架能力的小示例。通过这个项目,你可以学习到如何处理文件系统数据,实现基本的文件操作界面,以及如何根据不同的文件类型提供不同的功能。PyQt5为桌面应用开发提供了广泛的可能性,你可以在此基础上继续扩展功能,打造更加强大的应用程序。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

cad二次开发autolisp(二)

目录 一、选择集1.1 选择集的创建1.2 选择集的编辑1.3 操作选择集 二、命令行设置对话框2.1 设置图层2.2 加载线型2.3 设置字体样式2.4 设置标注样式(了解即可) 三、符号表3.1 简介3.2 符号表查找3.2 符号表删改增 一、选择集 定义:批量选择…

Springboot 项目启动成功,但是访问接口404

搭建了一个子项目运行,启动没有任何问题,但是访问restful接口404,查阅网上是启动类与包路径问题,刚开始按照网上说的是扫描不到,但是检测我创建的路径并没有问题,具体怎么确定是启动时在扫描创建beanDefini…

Java框架篇面试题

📕作者简介: 过去日记,致力于Java、GoLang,Rust等多种编程语言,热爱技术,喜欢游戏的博主。 📗本文收录于java面试题系列,大家有兴趣的可以看一看 📘相关专栏Rust初阶教程、go语言基…

需求分析的结构化和面向对象方法

需求分析的方法,主要有结构化(SA)方法、面向对象分析(OOA)方法。另外有面向问题域的分析(Problem Domain Oriented Analysis,PDOA)方法,还有一些形式化方法例如VDM ( Vienna Design Method)和Z等…

银行数据仓库体系实践(4)--数据抽取和加载

1、ETL和ELT ETL是Extract、Transfrom、Load即抽取、转换、加载三个英文单词首字母的集合: E:抽取,从源系统(Souce)获取数据; T:转换,将源系统获取的数据进行处理加工,比如数据格式转化、数据精…

AP5191 降压恒流 双灯 12V5A 一切一LED车灯汽车大灯驱动方案

AP5191是一款PWM工作模式,高效率、外围简 单、内置功率MOS管,适用于4.5-150V输入的高 精度降压LED恒流驱动芯片。输出功率150W, 电流6A。 AP5191可实现线性调光和PWM调光,线性调 光脚有效电压范围0.55-2.6V. AP5191 工作频率可以通过RT 外部…

Programming Abstractions in C阅读笔记:p248-p253

《Programming Abstractions in C》学习第69天,p248-p253总结,总计6页。 一、技术总结 “A generalized program for two-player games”如标题所示,该小节强调要学会从一个复杂的程序中抽象出通用的内容——这也是本书的主旨——“Program…

C语言进阶——数据结构之链表

前言 hello,大家好呀,我是Humble 在之前的两篇博客,我们学完了数据结构中的顺序表,还对它进行了一个应用,做了一个通讯录的小项目 那今天我们再来学习一个新的数据结构——链表 引入 我们来回忆一下顺序表 对于顺…

工业智能网关储能物联网应用实现能源的高效利用及远程管理

储能电力物联网是指利用物联网技术和储能技术相结合,实现对电力系统中各种储能设备的智能管理和优化控制。随着可再生能源的不断发展和应用,电力系统面临着越来越大的电力调度和储能需求而储能电力物联网的出现可以有效解决这一问题,提高电力…

领势 跨越|美创科技数据安全管理平台2023年度回顾

领势 跨越 数据安全管理平台的2023 以数字化安全平台为基础的数据安全体系是未来安全建设的重要方向。 研究咨询机构IDC指出:未来,数据安全基础设施管理平台将逐步发展成为各组织数据安全建设的基础设施。 Gartner《中国数据安全治理解析》预测&…

顶顶通呼叫中心中间件机器人压力测试配置(mod_cti基于FreeSWITCH)

介绍 顶顶通呼叫中心中间件机器人压力测试(mod_cit基于FreeSWITCH) 一、配置acl.conf 打开ccadmin-》点击配置文件-》点击acl.conf-》我这里是已经配置好了的,这里的192.168.31.145是我自己的内网IP,你们还需要自行修改 二、配置线路 打开ccadmin-&g…

Node.js Stream.pipeline() Method

Why Stream.pipeline 通过流我们可以将一大块数据拆分为一小部分一点一点的流动起来,而无需一次性全部读入,在 Linux 下我们可以通过 | 符号实现,类似的在 Nodejs 的 Stream 模块中同样也为我们提供了 pipe() 方法来实现。 未使用 Stream p…

解决 ssh: connect to host github.com port 22: Connection timed out

问题 今天使用git克隆github上的代码时,一直报错 原以为是公钥过期了,就尝试修改配置公钥,但是尝试了几次都不行,最终在博客上找到了解决方案,在次记录一下,以备不时之需 解决ssh-connect-to-host-github…

springboot整合MongoDB实战

目录 环境准备 引入依赖 配置yml 注入mongoTemplate 集合操作 文档操作 创建实体 添加文档 查询文档 更新文档 删除文档 环境准备 引入依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-da…

Flink(十五)【Flink SQL Connector、savepoint、CateLog、Table API】

前言 今天一天争取搞完最后这一部分&#xff0c;学完赶紧把 Kafka 和 Flume 学完&#xff0c;就要开始做实时数仓了。据说是应届生得把实时数仓搞个 80%~90% 才能差不多找个工作&#xff0c;太牛马了。 1、常用 Connector 读写 之前我们已经用过了一些简单的内置连接器&#x…

机器学习预测全家桶之单变量输入多步预测,天气温度预测为例,MATLAB代码

截止到本期&#xff0c;一共发了8篇关于机器学习预测全家桶的文章。参考文章如下&#xff1a; 1.五花八门的机器学习预测&#xff1f;一篇搞定不行吗&#xff1f; 2.机器学习预测全家桶&#xff0c;多步预测之BiGRU、BiLSTM、GRU、LSTM&#xff0c;LSSVM、TCN、CNN&#xff0c;…

怎么快速发表一篇EI会议论文?有什么要注意的?

都说EI会议论文的发表相对简单一些&#xff0c;但因为EI会议论文的含金量也挺高&#xff0c;因此很多国内外作者都喜爱在EI上投稿论文&#xff0c;那么怎么在国际ei会议发表会议论文呢? 这和国内发表论文都是差不多的&#xff0c;要选择合适的会议&#xff0c;按照会议要求整…

web安全学习笔记【06】——http\https抓包

思维导图放最后 #知识点&#xff1a; 1、Web常规-系统&中间件&数据库&源码等 2、Web其他-前后端&软件&Docker&分配站等 3、Web拓展-CDN&WAF&OSS&反向&负载均衡等 ----------------------------------- 1、APP架构-封装&原生态&…

Textual Inversion、DreamBooth、LoRA、InstantID:从低成本进化到零成本实现IP专属的AI绘画模型

2023年7月份国内有一款定制写真AI工具爆火。一款名为妙鸭相机的AI写真小程序&#xff0c;成功在C端消费者群体中出圈&#xff0c;并在微信、微博和小红书等平台迅速走红&#xff0c;小红书上的话题Tag获得了330多万的浏览量&#xff0c;相关微信指数飙升到了1800万以上。 其他…

【RT-DETR有效改进】2023.12月份最新成果TransNeXt像素聚焦注意力主干(全网首发)

前言 大家好&#xff0c;我是Snu77&#xff0c;这里是RT-DETR有效涨点专栏。 本专栏的内容为根据ultralytics版本的RT-DETR进行改进&#xff0c;内容持续更新&#xff0c;每周更新文章数量3-10篇。 专栏以ResNet18、ResNet50为基础修改版本&#xff0c;同时修改内容也支持Re…