头图

介绍

HarmonyOS的沙盒机制完全屏蔽了应用对手机公共存储空间的访问,安全性提高已不言而喻。 本篇文章的主要目的是为了能通过一个简单工具,可视化的让一个新手能相对轻松的学习文件&数据存储。HarmonyOS 应用开发工具DevEco Studio也没有提供读取存储器的功能,所以做一个简单版本的应用级别浏览器还是有必要的。

准备

  1. 请参照[官方指导],创建一个Demo工程,选择Stage模型
  2. 须熟读HarmonyOS 指导“[文件管理]gitee.com/li-shizhen-skin/harmony-os/blob/master/README.md

V2.0白皮书.png

效果

页面由三部分组成:1.标题栏(用于返回)2. 当前路径描述 3. 文件列表

Screenshot_20231204204057250.png

应用文件目录结构图

如果之前你熟悉Android存储目录,在学习HarmonyOS时,可以忘记之前所有了的东西了,因为HarmonyOS真的能让研发人员感受沙盒的存在,这点可以对标iOS系统

0000000000011111111.20231121184039.03067481202577079515515889587732.png

代码片段解析

1)获取个人应用沙盒下的所有根目录

this.context.bundleCodeDir
this.context.cacheDir
this.context.filesDir
this.context.preferencesDir
this.context.tempDir
this.context.databaseDir
this.context.distributedFilesDir

2)遍历目录

import fs from '@ohos.file.fs';

//遍历 this.currentRoot 路径下的所有文件和文件夹
fs.listFile(this.currentRoot)

3)文件/文件夹类型判断

try {
  stat = fs.statSync(tempPath)
} catch (e) {
  //路径可能不存在
  return
}

//判断是否为文件夹
stat.isDirectory()

//判断是否为文件
stat.isFile()

核心文件

import fs from '@ohos.file.fs';
import common from '@ohos.app.ability.common';
import { CommonConstants } from '../common/CommonConstants';
import FileManagerStructModel from '../viewmodel/FileManagerStruct';
import { Title } from '../view/Title';
import router from '@ohos.router';

/**
 * 官方指导
 * https://developer.harmonyos.com/cn/docs/documentation/doc-guides-V3/application-context-stage-0000001427744560-V3#ZH-CN_TOPIC_0000001574128741__%E8%8E%B7%E5%8F%96%E5%BA%94%E7%94%A8%E5%BC%80%E5%8F%91%E8%B7%AF%E5%BE%84
 *
 */
@Entry
@Component
struct FileManagerIndex {
  private context = getContext(this) as common.UIAbilityContext;

  //当前路径下的文件和文件夹
  @State listData: FileManagerStructModel[] = []

  @State filePaths: string[] = []
  @State childFilePaths: string[] = []
  @State currentRoot: string = ''
  pathLevel: number = 0

  @State pathTips: string = ''

  scroller: Scroller = new Scroller();

  initItem(item){
    let model = new FileManagerStructModel()
    model.currentDirName = item.substring(item.lastIndexOf('/')+1)
    model.currentPath = item.substring(0, item.lastIndexOf('/'))

    let stat

    try {
      stat = fs.statSync(item)
    } catch (e){
      console.log(e)
      return
    }

    if(fs.statSync(item).isDirectory()){
      model.isDir = true
      model.smallIMG = $r('app.media.ic_files_folder')
    } else {
      model.isDir = false
      if(item.toLowerCase().endsWith('.png')
        || item.toLowerCase().endsWith('.png')
        || item.toLowerCase().endsWith('.jpg')
        || item.toLowerCase().endsWith('.bmp')){
        model.smallIMG = $r('app.media.ic_files_image')
      } else  if(item.toLowerCase().endsWith('.doc')) {
        model.smallIMG = $r('app.media.ic_files_doc')
      } else  if(item.toLowerCase().endsWith('.log')) {
        model.smallIMG = $r('app.media.ic_files_log')
      } else  if(item.toLowerCase().endsWith('.html')) {
        model.smallIMG = $r('app.media.ic_files_html')
      } else  if(item.toLowerCase().endsWith('.pdf')) {
        model.smallIMG = $r('app.media.ic_files_pdf')
      } else  if(item.toLowerCase().endsWith('.xml')) {
        model.smallIMG = $r('app.media.ic_files_xml')
      } else  if(item.toLowerCase().endsWith('.txt')) {
        model.smallIMG = $r('app.media.ic_files_grid_txt')
      } else  if(item.toLowerCase().endsWith('.chm')) {
        model.smallIMG = $r('app.media.ic_files_chm')
      } else  if(item.toLowerCase().endsWith('.pptx')) {
        model.smallIMG = $r('app.media.ic_files_pptx')
      } else  if(item.toLowerCase().endsWith('.mp3')) {
        model.smallIMG = $r('app.media.ic_files_mp3')
      } else  if(item.toLowerCase().endsWith('.mp4')) {
        model.smallIMG = $r('app.media.ic_files_m4a')
      } else {
        model.smallIMG = $r('app.media.ic_files_unknown')
      }
    }

    this.listData.push(model)
  }

  initCurrentDir(){
    this.pathTips = '我的应用'

    this.listData = []
    this.filePaths.forEach((item)=>{
       this.initItem(item)
    })
  }

  aboutToAppear(){
    this.filePaths.push(this.context.bundleCodeDir)
    this.filePaths.push(this.context.cacheDir)
    this.filePaths.push(this.context.filesDir)
    this.filePaths.push(this.context.preferencesDir)
    this.filePaths.push(this.context.tempDir)
    this.filePaths.push(this.context.databaseDir)
    this.filePaths.push(this.context.distributedFilesDir)

    console.log(this.context.bundleCodeDir)
    console.log(this.context.cacheDir)
    console.log(this.context.filesDir)
    console.log(this.context.preferencesDir)
    console.log(this.context.tempDir)
    console.log(this.context.databaseDir)
    console.log(this.context.distributedFilesDir)

    this.initCurrentDir()

  }

  build(){
    Column(){
      Title({title: '返回'})
      .onClick(()=> {
        if (this.currentRoot != this.context.bundleCodeDir
        && this.currentRoot != this.context.cacheDir
        && this.currentRoot != this.context.filesDir
        && this.currentRoot != this.context.preferencesDir
        && this.currentRoot != this.context.tempDir
        && this.currentRoot != this.context.databaseDir
        && this.currentRoot != this.context.distributedFilesDir) {

          this.currentRoot = this.currentRoot.substring(0, this.currentRoot.lastIndexOf('/'))

          this.pathTips = this.pathTips.substring(0, this.pathTips.lastIndexOf('>'))

          this.pathLevel--
          fs.listFile(this.currentRoot)
            .then((filesPath: string[]) => {
              this.listData = []
              filesPath.forEach((info) => {
                this.initItem(this.currentRoot + '/' + info)
              })
            })

        } else {

          this.pathLevel--

          console.log('当前级数:' + this.pathLevel)

          if(this.pathLevel < 0){
            router.back()
            return
          } else {
            this.initCurrentDir()
          }

        }
      })
      .height('5%')

      Text(this.pathTips).fontSize(10).fontColor(Color.Gray).width('100%').margin({top: 10, bottom: 10})

      Stack(){
        if(this.listData == null || this.listData.length == 0){
          Text('空空如也')
            .width('100%')
            .height('100%')
            .fontSize(25)
            .textAlign(TextAlign.Center)
        } else {
          List(){
            ForEach(this.listData, (item: FileManagerStructModel, index: number) => {
              ListItem(){
                Row(){
                  Image(item.smallIMG).width(30).height(30).objectFit(ImageFit.Contain)
                  Text(item.currentDirName).margin({left: 10})
                }.width('100%').height(60).onClick(()=>{

                  let tempPath = item.currentPath + '/' + item.currentDirName
                  console.log(tempPath)

                  let stat

                  try {
                    stat = fs.statSync(tempPath)
                  } catch (e) {
                    console.log(e)
                    return
                  }

                  if(stat.isDirectory()){
                    this.currentRoot = tempPath

                    this.pathTips = this.pathTips + '>' + item.currentDirName;

                    this.pathLevel++

                    this.listData = []
                    fs.listFile(this.currentRoot)
                      .then( (filePaths: string[]) => {
                          filePaths.forEach((info)=>{
                            this.initItem(this.currentRoot+'/'+info)
                          }
                        )
                      })
                  }

                })
              }
            })
          }
          .width('100%')
          .height('100%')
        }
      }
      .width('100%')
      .height('94%')
    }
    .alignItems(HorizontalAlign.Start)
    .margin({top: 10})
    .padding({top: px2vp(AppStorage.Get(CommonConstants.StatusBarHeight)), left: '10vp', right: '10vp'})

  }

}

烧脑猴
24 声望21 粉丝

5年JAVA,3年Android。现转入鸿蒙开发行业,每日分享一些鸿蒙技术!