news 2026/4/18 14:16:31

开源鸿蒙跨平台开发训练营--AtomGit(GitCode)口袋工具(六)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
开源鸿蒙跨平台开发训练营--AtomGit(GitCode)口袋工具(六)

获取代码仓库文件列表

如下图所示:我们先实现从网络获取文件列表的接口

该接口会返回文件列表集合。实例数据如下所示:

[ .gitignore, ArkTSCangjieHybridApp/.gitignore, ArkTSCangjieHybridApp/AppScope/resources/base/element/string.json, ArkTSCangjieHybridApp/AppScope/resources/base/media/app_icon.png, ArkTSCangjieHybridApp/README.md, ArkTSCangjieHybridApp/arkts_modules/shortvideoarkts/.gitignore, ArkTSCangjieHybridApp/arkts_modules/shortvideoarkts/Index.ets, ArkTSCangjieHybridApp/arkts_modules/shortvideoarkts/oh-package.json5, ]

1. 数据处理

首先我们需要创建一个GitCodeCodeRepo实体类,用来表示文件或者文件夹。根据上述返回的是数据列表,我们需要将每个文件、文件夹转换成对应的GitCodeCodeRepo实体类。所以我们需要做数据转换,将对应的字符串列表,转换成List<GitCodeCodeRepo>

GitCodeCodeRepo我们先设计三个属性,如下所示:

  • name:文件名/文件夹名。
  • path:文件夹/文件的完整路径。
  • isDir: 是否是文件夹。
class GitCodeCodeRepo { final String name; // 文件名或文件夹名 final bool isDir; // 是否为文件夹 final String? path; // 完整路径 GitCodeCodeRepo({ required this.name, required this.isDir, this.path, }); }

然后我们在GitCodeCodeRepo中新建一个buildTreeFromPaths函数,专门用来解析数据,目的是将List<String>转换成List<GitCodeCodeRepo>。代码如下所示:

static List<GitCodeCodeRepo> buildTreeFromPaths(List<String> paths) { final List<GitCodeCodeRepo> rootNodes = []; // 遍历接口返回的所有文件路径 for (final path in paths) { // 将每个路径按照 "/" 进行分割。 final parts = path.split('/').where((part) => part.isNotEmpty).toList(); if (parts.isEmpty) continue; for (int i = 0; i < parts.length; i++) { // 获取文件名 String fileName = parts[i]; // 判断是不是文件夹 bool isDir = fileName.contains("."); GitCodeCodeRepo repo = GitCodeCodeRepo(name: fileName, isDir: isDir); rootNodes.add(repo); } } return rootNodes; }

文件和文件夹虽然显示出来了,但是我们发现数据有问题:所有的文件和文件夹都平铺在这个列表里,没有文件夹层次结构了。‘

因此我们继续修改,并且让当前列表只展示根目录下的文件或文件夹。

static List<GitCodeCodeRepo> buildTreeFromPathsTest(List<String> paths) { final Map<String, GitCodeCodeRepo> nodeMap = {}; final List<GitCodeCodeRepo> rootNodes = []; // 遍历接口返回的所有文件路径 for (final path in paths) { // 将每个路径按照 "/" 进行分割。 final parts = path.split('/').where((part) => part.isNotEmpty).toList(); if (parts.isEmpty) continue; String currentPath = ''; for (int i = 0; i < parts.length; i++) { final part = parts[i]; final isLastPart = i == parts.length - 1; // 这里修改了判断文件夹的逻辑 final isDir = isLastPart ? path.endsWith('/') : true; currentPath += '/$part'; // 对于文件夹,确保路径以斜杠结尾 final nodePath = isDir ? '$currentPath/' : currentPath; if (!nodeMap.containsKey(nodePath)) { final node = GitCodeCodeRepo( name: part, isDir: isDir, path: nodePath, ); nodeMap[nodePath] = node; // 如果是根节点(第一级),添加到根节点列表 if (i == 0) { rootNodes.add(node); } } } } return rootNodes; }

说明:为了提高用户体验,区分不同的 item。我们对每个 item 进行各行设置不同的背景色。

根目录显示成功,我们可以继续再优化一下。当前处理后的列表中,文件夹和文家是乱序的,我们需要将文件夹和文件进行排序,如果相同类型的文件或文件夹,则按照字母顺序排序。

// 在该函数中, return List 之前,先进行排序 rootNodes.sort((a, b) { // 文件夹排在前面 if (a.isDir && !b.isDir) return -1; if (!a.isDir && b.isDir) return 1; // 同类型按名称字母顺序排序 return a.name.compareTo(b.name); });

2. 界面跳转

当我们点击 item 后跳转到文件详情界面。目前先展示一些基本信息。文件具体的内容暂时先不展示。

// 点击跳转 Navigator.push( context, MaterialPageRoute( builder: (context) => ShowCodeDetailPage( repo: widget.repo, refName: refName, filePath: fullPath, node: node, fileTree: _fileTree, // 传递_fileTree数据 ), ), );

在文件详情界面我们展示一些基本信息,用于验证文件的正确性。如下图所示:仓库、分支正确,而且最重要的文件路径也显示正确。

3. 侧边栏实现

由于文件夹和文件列表以及文件具体内容展示区域都非常占用空间,所以我们将文件夹和文件列表使用侧边栏来展示,并且侧边栏是悬浮状态。而且由于我们将处理过后的文件列表数据传递过去了,也顺便在侧边栏中展示出来。

核心区域我们做一个判断,如果用户点击的 item 是文件夹进入当前页面,则核心区域不展示内容。如果用户点击的是 item 是文件,则展示当前文件内容(先不实现)。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/17 6:19:23

draw.io 默认字体太丑?用这一招瞬间变高级!

背景 draw.io 是一款简洁、高效的画图工具&#xff0c;但其默认可选字体较为有限&#xff1a;如果你想在图表中使用自己喜欢的字体&#xff08;如第三方中文字体&#xff09;&#xff0c;往往会发现列表里压根找不到。要解决这一问题&#xff0c;必须先了解&#xff1a;draw.io…

作者头像 李华
网站建设 2026/4/17 17:07:41

领域驱动设计:构建业务与技术的桥梁

领域驱动设计&#xff1a;构建业务与技术的桥梁 【免费下载链接】geektime-books :books: 极客时间电子书 项目地址: https://gitcode.com/GitHub_Trending/ge/geektime-books 在数字化浪潮席卷各行各业的今天&#xff0c;我们面临着一个共同的挑战&#xff1a;如何在快…

作者头像 李华
网站建设 2026/4/18 3:26:08

办理无线充 FCC 认证需要准备什么材料?

无线充办理 FCC 认证需区分具体类型&#xff0c;若为仅含电磁感应的普通无线充&#xff08;无主动射频发射&#xff09;&#xff0c;走SDoC模式&#xff1b;若集成蓝牙 / Wi-Fi 等无线通信功能&#xff0c;则需走FCC ID模式&#xff0c;两类认证所需材料有明确差异&#xff0c;…

作者头像 李华
网站建设 2026/4/18 3:25:29

算法题 设计哈希集合

设计哈希集合 问题描述 不使用任何内建的哈希表库设计一个哈希集合&#xff08;HashSet&#xff09;。 实现 MyHashSet 类&#xff1a; void add(key) 向哈希集合中插入一个值 key。bool contains(key) 返回哈希集合中是否包含这个值 key。void remove(key) 将给定值 key 从哈希…

作者头像 李华
网站建设 2026/4/17 22:41:02

如何快速掌握自主移动机器人:从入门到实战的完整指南

如何快速掌握自主移动机器人&#xff1a;从入门到实战的完整指南 【免费下载链接】划重点自主移动机器人导论.pdf资源介绍 《自主移动机器人导论.pdf》是一本系统梳理自主移动机器人知识的实用指南&#xff0c;涵盖基本概念、技术原理、发展历程及应用前景等内容。本书语言通俗…

作者头像 李华
网站建设 2026/4/18 3:26:04

C++并发编程工作窃取算法:彻底搞懂memory_order_acquire/release

案例它实现了一个基于**工作窃取算法(Work-Stealing Algorithm)**的线程池系统,这是一种优雅而高效的动态负载均衡策略。其核心思想简单而深刻:当一个线程完成了自己的任务后,它不会闲着,而是会主动去"窃取"其他仍在忙碌的线程的任务来执行。这种机制确保了所有…

作者头像 李华