鸿蒙 Next 实战:打造全能文件管理器应用

news/2024/11/6 9:34:52/文章来源:https://www.cnblogs.com/samex/p/18529283

本文旨在深入探讨华为鸿蒙HarmonyOS Next系统(截止目前 API12)在开发多语言电商平台方面的技术细节,基于实际开发实践进行总结。主要作为技术分享与交流载体,难免错漏,欢迎各位同仁提出宝贵意见和问题,以便共同进步。本文为原创内容,任何形式的转载必须注明出处及原作者。

在当今数字化时代,文件管理器成为了人们管理本地文件和媒体文件的重要工具。今天,我们将基于鸿蒙 Next 系统,深入探讨如何开发一个功能强大的文件管理器应用,涵盖从基础架构设计到核心功能实现的全过程,让你全面掌握鸿蒙 Next 在文件管理领域的应用开发技巧。

一、应用架构设计:MVC 架构的应用

(一)MVC 架构概述

MVC(Model-View-Controller)架构是一种经典的软件设计模式,它将应用程序分为三个主要部分:模型(Model)、视图(View)和控制器(Controller)。在我们的文件管理器应用中,这种架构的应用将使代码结构更加清晰,易于维护和扩展。

(二)模型(Model)层

模型层负责处理数据的存储、检索和更新。在文件管理器应用中,它主要涉及文件和目录的操作,如文件读取、写入、删除、复制、移动等。我们将使用鸿蒙 Next 的系统 API 来实现这些操作。例如,使用fileIo模块来进行文件的读写操作,directoryIo模块来处理目录相关操作。

(三)视图(View)层

视图层负责呈现用户界面,展示文件和目录列表,以及提供操作按钮等交互元素。我们将使用 ArkUI 框架来构建用户界面,通过组件化的方式,如List组件展示文件列表,Button组件实现操作按钮,确保界面简洁、美观且易于操作。

(四)控制器(Controller)层

控制器层充当模型和视图之间的桥梁,负责处理用户输入事件,并根据业务逻辑调用模型层的相应方法。例如,当用户点击文件上传按钮时,控制器将获取用户选择的文件路径,调用模型层的上传方法将文件上传到指定位置。

(五)使用系统 Picker 实现文件选择和保存

为了遵循鸿蒙 Next 的安全原则,避免直接访问用户文件系统,我们将广泛使用系统 Picker 来实现文件选择和保存功能。当用户需要打开文件时,通过文件选择器(FilePicker)让用户选择文件,应用获取用户选择的文件路径后进行后续操作。同样,在保存文件时,使用系统提供的保存路径选择器,确保文件保存到正确的位置。

二、权限申请与管理

(一)权限机制回顾

在鸿蒙 Next 系统中,权限分为 system_grant(系统授权)和 user_grant(用户授权)两种类型。系统授权权限在应用安装时自动授予,而用户授权权限则需要在应用运行时向用户请求授权。

(二)应用所需权限及申请方式

  1. 读取文件权限(user_grant)
    应用需要读取用户本地文件时,需申请ohos.permission.READ_EXTERNAL_STORAGE权限(假设为读取外部存储文件权限,实际根据系统定义)。在应用启动时,通过requestPermissionsFromUser()接口向用户请求授权。例如:
import { abilityAccessCtrl, common, Permissions } from '@kit.AbilityKit';
import { BusinessError } from '@kit.BasicServicesKit';const readPermission: Permissions = 'ohos.permission.READ_EXTERNAL_STORAGE';async function checkReadPermissionGrant(): Promise<abilityAccessCtrl.GrantStatus> {let atManager: abilityAccessCtrl.AtManager = abilityAccessCtrl.createAtManager();let grantStatus: abilityAccessCtrl.GrantStatus = abilityAccessCtrl.GrantStatus.PERMISSION_DENIED;// 获取应用程序的 accessTokenIDlet tokenId: number = 0;try {let bundleInfo: bundleManager.BundleInfo = await bundleManager.getBundleInfoForSelf(bundleManager.BundleFlag.GET_BUNDLE_INFO_WITH_APPLICATION);let appInfo: bundleManager.ApplicationInfo = bundleInfo.appInfo;tokenId = appInfo.accessTokenId;} catch (error) {const err: BusinessError = error as BusinessError;console.error(`Failed to get bundle info for self. Code is ${err.code}, message is ${err.message}`);}// 校验应用是否被授予权限try {grantStatus = await atManager.checkAccessToken(tokenId, readPermission);} catch (error) {const err: BusinessError = error as BusinessError;console.error(`Failed to check access token. Code is ${err.code}, message is ${err.message}`);}return grantStatus;
}async function requestReadPermission(): Promise<void> {let grantStatus: abilityAccessCtrl.GrantStatus = await checkReadPermissionGrant();if (grantStatus === abilityAccessCtrl.GrantStatus.PERMISSION_GRANTED) {// 已获得读取权限,可以进行文件读取操作console.log('已获得读取文件权限,可以继续操作。');} else {// 申请读取文件权限const permissions: Array<Permissions> = [readPermission];let atManager: abilityAccessCtrl.AtManager = abilityAccessCtrl.createAtManager();atManager.requestPermissionsFromUser(globalThis.context as common.UIAbilityContext, permissions).then((data) => {let grantStatus: Array<number> = data.authResults;let length: number = grantStatus.length;for (let i = 0; i < length; i++) {if (grantStatus[i] === 0) {// 用户授权,可以继续访问目标操作console.log('用户已授权读取文件权限,可以继续操作。');} else {// 用户拒绝授权,提示用户必须授权才能访问相关功能,并引导用户到系统设置中打开相应权限console.log('用户拒绝授权读取文件权限,请前往系统设置中手动授予权限。');return;}}// 授权成功}).catch((err: BusinessError) => {console.error(`Failed to request read permission from user. Code is ${err.code}, message is ${err.message}`);});}
}// 在应用启动或需要读取文件时调用 requestReadPermission() 函数
requestReadPermission();
  1. 写入文件权限(user_grant)
    当应用需要保存文件或修改文件内容时,需申请ohos.permission.WRITE_EXTERNAL_STORAGE权限。申请方式与读取文件权限类似,同样需要先检查权限状态,未授权时向用户请求授权。

  2. 访问网络权限(system_grant)
    文件上传和下载功能需要访问网络,应用需申请ohos.permission.INTERNET权限。此权限为系统授权类型,在应用的配置文件(如module.json5)中声明即可,系统会在安装时自动授予。

(三)受限开放权限与 ACL 申请(以读取媒体文件为例)

如果应用需要读取用户媒体文件,如图片、音频、视频等,可能涉及受限开放权限。假设读取媒体文件权限为ohos.permission.READ_MEDIA_FILES(实际根据系统定义),且该权限属于受限开放权限。

  1. AGC 申请 Profile 文件
    首先,开发者需要在应用市场(AppGallery Connect,AGC)申请 Profile 文件,并在申请过程中明确申请使用读取媒体文件权限。提供详细的应用使用场景说明,如文件管理器需要读取媒体文件以实现预览、分类管理等功能。

  2. 代码工程中声明权限
    在 AGC 侧申请成功后,在代码工程的配置文件(module.json5)中声明该权限:

{
"module": {
"requestPermissions":[
{
"name": "ohos.permission.READ_MEDIA_FILES",
"reason": "$string:reason_for_read_media_files",
"usedScene": {
"abilities": [
"MainAbility"
],
"when":"inuse"
}
}
]
}
}

同时,在应用运行时,按照用户授权的流程,通过requestPermissionsFromUser()接口向用户请求授权(如果该权限属于 user_grant 类型),并处理用户的授权结果。

三、核心功能实现

(一)文件操作功能

  1. 文件读取
    使用fileIo模块的open()方法打开文件,获取文件描述符,然后通过read()方法读取文件内容。例如:
import { fileIo } from '@kit.CoreFileKit';async function readFile(filePath: string): Promise<string> {let file = await fileIo.open(filePath, fileIo.OpenMode.READ_ONLY);let buffer = new ArrayBuffer(fileIo.statSync(filePath).size);await fileIo.read(file.fd, buffer);await fileIo.close(file.fd);return new TextDecoder().decode(buffer);
}
  1. 文件写入
    通过fileIo模块的open()方法以写入模式打开文件,使用write()方法将数据写入文件。例如:
async function writeFile(filePath: string, content: string): Promise<void> {let file = await fileIo.open(filePath, fileIo.OpenMode.WRITE_ONLY | fileIo.OpenMode.CREATE);await fileIo.write(file.fd, new TextEncoder().encode(content));await fileIo.close(file.fd);
}
  1. 文件删除、复制和移动
    利用fileIodirectoryIo模块的相关方法实现文件的删除、复制和移动操作。例如,文件删除可以使用unlink()方法,文件复制可以先读取源文件内容,再写入到目标文件,文件移动可以通过先复制再删除源文件的方式实现(需考虑原子性操作以确保数据完整性)。

(二)网络操作功能

  1. 文件上传
    使用httphttps模块(假设鸿蒙 Next 提供类似网络请求模块)实现文件上传功能。首先创建一个HttpRequest对象,设置请求方法为POST,上传地址等参数,然后将文件内容作为请求体发送。例如:
import { http } from '@kit.NetworkKit';async function uploadFile(filePath: string, uploadUrl: string): Promise<void> {let fileContent = await readFile(filePath);let request = http.createHttpRequest();request.method = 'POST';request.url = uploadUrl;request.headers = { 'Content-Type': 'application/octet-stream' };request.requestBody = new Uint8Array(new TextEncoder().encode(fileContent));try {await request.send();if (request.responseCode === 200) {console.log('文件上传成功。');} else {console.error('文件上传失败,错误码:', request.responseCode);}} catch (error) {console.error('文件上传过程中出错:', error);}
}
  1. 文件下载
    类似地,使用网络请求模块实现文件下载功能。创建HttpRequest对象,设置请求方法为GET,下载地址等参数,然后接收服务器返回的文件内容并保存到本地。例如:
async function downloadFile(downloadUrl: string, savePath: string): Promise<void> {let request = http.createHttpRequest();request.method = 'GET';request.url = downloadUrl;try {await request.send();if (request.responseCode === 200) {await writeFile(savePath, request.responseData.toString());console.log('文件下载成功,保存路径:', savePath);} else {console.error('文件下载失败,错误码:', request.responseCode);}} catch (error) {console.error('文件下载过程中出错:', error);}
}

(三)分享功能

  1. 分享方式选择
    提供多种分享方式,如通过邮件、短信、社交媒体等分享文件。在用户选择分享文件后,弹出分享方式选择界面,让用户选择合适的分享渠道。

  2. 分享实现
    根据用户选择的分享方式,使用相应的系统 API 实现分享操作。例如,通过邮件分享文件时,使用邮件客户端的分享接口,将文件作为附件添加到邮件中。假设鸿蒙 Next 提供了share模块来实现分享功能,以下是一个简单的邮件分享示例(实际接口可能不同):

import { share } from '@kit.ShareKit';async function shareFileByEmail(filePath: string, recipient: string): Promise<void> {let fileContent = await readFile(filePath);let shareData = {type:'message/email',subject: '文件分享',body: '这是一个通过文件管理器分享的文件。',attachments: [{name: filePath.split('/').pop(),data: fileContent}]};try {await share.share(shareData, { recipients: [recipient] });console.log('文件通过邮件分享成功。');} catch (error) {console.error('文件分享失败:', error);}
}

四、总结与展望

通过本次实战,我们成功构建了一个基于鸿蒙 Next 系统的文件管理器应用,涵盖了文件浏览、管理、上传、下载和分享等核心功能。在开发过程中,我们深入应用了鸿蒙 Next 的应用沙箱与权限机制、系统授权与用户授权、受限开放权限与 ACL 申请、系统 Picker 等关键技术,确保了应用的安全性、稳定性和功能性。

展望未来,随着鸿蒙 Next 系统的不断发展和完善,我们可以进一步优化文件管理器的性能,如提升大文件操作的效率、增强文件搜索功能等。同时,结合分布式技术,实现跨设备的文件管理和共享,为用户提供更加便捷、高效的文件管理体验。希望本文能够为鸿蒙 Next 同行者提供有益的参考和借鉴,激发更多创新应用的开发。

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Nacos原理汇总

今天就应某位小伙伴的要求,来讲一讲Nacos作为服务注册中心底层的实现原理不知你是否跟我一样,在使用Nacos时有以下几点疑问:临时实例和永久实例是什么?有什么区别? 服务实例是如何注册到服务端的? 服务实例和服务端之间是如何保活的? 服务订阅是如何实现的? 集群间数据…

人工智能模型训练中的数据之美——探索TFRecord

上一篇:《构建人工智能模型基础:TFDS和Keras的完美搭配》 序言:在人工智能模型的训练过程中,如何高效管理和处理大量数据是一个重要的课题。TensorFlow 的 TFRecord 格式为大规模数据存储和处理提供了一种灵活且高效的解决方案。在本节知识中,我们将介绍如何利用 TFRecord…

教师提前批试讲-注意事项

教师提前批试讲-注意事项

大白菜装系统

在平时工作中,作为程序员,最苦逼的是公司的电脑坏了,都找你。我想说我是程序员,不是修电脑的,但是架不住小姐姐的热情,还是做了。 装系统流程: 第一步:前期准备 1、使用【大白菜】制作U盘启动盘。 2、查询机型的U盘启动快捷键。 3、准备一个ISO/GHO镜像。 第二步:插入…

实验11:装饰模式

本次实验属于模仿型实验,通过本次实验学生将掌握以下内容: 1、理解装饰模式的动机,掌握该模式的结构; 2、能够利用装饰模式解决实际问题。[实验任务一]:手机功能的升级用装饰模式模拟手机功能的升级过程:简单的手机(SimplePhone)在接收来电时,会发出声音提醒主人;而Ja…

Pbootcms网站,从Apache切换为Nginx后网站打不开

打开网站设置登录宝塔面板。 选择需要配置的网站,点击“设置”。进入伪静态设置在网站设置页面中,找到并点击“伪静态”选项卡。添加Nginx伪静态规则在伪静态设置中,清空原有规则或选择自定义规则。 输入以下Nginx伪静态规则:location / {if (!-e $request_filename){rewri…

2024/11/06

软件设计 实验10:组合模式 用透明组合模式实现教材中的“文件夹浏览”这个例子。 类图 public abstract class AbstractFile {public abstract void add(AbstractFile element);public abstract void remove(AbstractFile element);public abstract void display(int depth); …

SQLSTATE[HY000] [1045] Access denied for user ‘root‘@‘localhost‘ (using password: YES)

错误解析错误代码:SQLSTATE[HY000] [1045] 错误信息:Access denied for user ‘root’@‘localhost’ (using password: YES)可能的原因密码错误:提供的密码与数据库中存储的密码不匹配。 用户权限问题:用户root可能没有从localhost访问数据库的权限。 配置文件问题:MySQL…

帝国CMS更改域名后信息地址中的域名不变解决方法

修改系统参数设置进入“系统参数设置 - 基本属性 - 网站地址”,设置为 /。 不要填写具体的域名,以避免信息地址变成绝对地址。更新信息页地址进入“系统 - 数据更新 - 更新信息页地址”,选择相应的数据表,点击“扫码添加技术【解决问题】专注中小企业网站建设、网站安全12年…

帝国CMS系统后台批量删除不带标题图片信息

执行SQL语句删除不带标题图片的信息:delete from [表前缀]_ecms_news where titlepic = ; delete from [表前缀]_ecms_news_index where id not in (select id from [表前缀]_ecms_news); delete from [表前缀]_ecms_news_data_1 where id not in (select id from [表前缀]_ec…

帝国CMS留言模板提示请输入姓名,邮箱与留言内容解决方法

检查表单提交:确认提交地址、POST 数据和表单字段正确。 在表单中添加隐藏字段 bid: <input name="bid" type="hidden" value="分类ID" />扫码添加技术【解决问题】专注中小企业网站建设、网站安全12年。熟悉各种CMS,精通PHP+MYSQL、HT…

信息安全新篇章:车企图纸安全外发创新方案!

车企的图纸在其业务流程中会与多个环节产生交集,并涉及到图纸安全外发传输。以下是主要涉及的环节及图纸传输发送的相关情况: 1.市场分析报告、概念草图、初步布置图等图纸:需要在设计团队内部进行传输,以供讨论和修改。同时,需要发送给供应商或合作伙伴进行初步评估和反馈…