1# Developing a File Manager Application (for System Applications Only) 2 3OpenHarmony is prebuilt with the **FileManager** application. You can also develop your own file manager application as required. 4 5## How to Develop 6For details about the APIs used to develop a file manager, see [User File Access and Management](../reference/apis/js-apis-fileAccess.md). 7 81. Apply for permissions required.<br> 9 Apply for the ohos.permission.FILE_ACCESS_MANAGER and ohos.permission.GET_BUNDLE_INFO_PRIVILEGED permissions. For details, see [Requesting Permissions for system_basic Applications](../security/AccessToken/determine-application-mode.md#requesting-permissions-for-system_basic-applications). 10 11 > **NOTE** 12 > 13 > - **ohos.permission.FILE_ACCESS_MANAGER** allows your application to use the user file access framework APIs. 14 >- **ohos.permission.GET_BUNDLE_INFO_PRIVILEGED** allows your application to obtain information about file management server applications supported by the system. 15 162. Import dependent modules. 17 18 ```ts 19 import fileAccess from '@ohos.file.fileAccess'; 20 import fileExtensionInfo from '@ohos.file.fileExtensionInfo'; 21 ``` 22 23 The **fileAccess** module provides APIs for basic file operations, and the **fileExtensionInfo** module provides key structs for application development. 24 253. Query device information.<br> 26 You can obtain attributes of the devices managed by one or all file management servers in the system. You can also filter devices as required. 27 28 In the user file access framework, **RootInfo** indicates the attribute information of a device. For example, obtain **RootInfo** of all devices. 29 30 ```ts 31 import common from '@ohos.app.ability.common'; 32 import { BusinessError } from '@ohos.base'; 33 import { Filter } from '@ohos.file.fs'; 34 35 // Obtain the application context. 36 let context = getContext(this) as common.UIAbilityContext; 37 38 // Create a helper object for connecting to all file management servers in the system. 39 let fileAccessHelperAllServer: fileAccess.FileAccessHelper; 40 function createFileAccessHelper(): void { 41 try { // this.context is the context passed from EntryAbility. 42 fileAccessHelperAllServer = fileAccess.createFileAccessHelper(context); 43 if (!fileAccessHelperAllServer) { 44 console.error("createFileAccessHelper interface returns an undefined object"); 45 } 46 } catch (err) { 47 let error: BusinessError = err as BusinessError; 48 console.error("createFileAccessHelper failed, errCode:" + error.code + ", errMessage:" + error.message); 49 } 50 } 51 let rootInfos: Array<fileAccess.RootInfo> = []; 52 async function getRoots(): Promise<void>{ 53 let rootIterator: fileAccess.RootIterator; 54 let isDone: boolean = false; 55 try { 56 rootIterator = await fileAccessHelperAllServer.getRoots(); 57 if (!rootIterator) { 58 console.error("getRoots interface returns an undefined object"); 59 return; 60 } 61 while (!isDone) { 62 let result = rootIterator.next(); 63 console.info("next result = " + JSON.stringify(result)); 64 isDone = result.done; 65 if (!isDone) 66 rootInfos.push(result.value); 67 } 68 } catch (err) { 69 let error: BusinessError = err as BusinessError; 70 console.error("getRoots failed, errCode:" + error.code + ", errMessage:" + error.message); 71 } 72 } 73 ``` 74 754. View directories.<br> 76 In the user file access framework, **FileInfo** indicates basic information about a file or folder. You can use **listfile()** to obtain a **FileIterator** object that traverses all files (folders) of the next level or use **scanfile()** to obtain a **FileIterator** object that meets the specified conditions. 77 78 Currently, **listfile()** and **scanfile()** can be called by the **RootInfo** object to traverse the next-level files or filter the entire directory tree. In addition, **listfile()** and **scanfile()** can be called by the **FileInfo** object to traverse the next-level files or filter the specified directories. 79 80 ```ts 81 import { BusinessError } from '@ohos.base'; 82 import { Filter } from '@ohos.file.fs'; 83 84 // Start from the root directory. 85 let rootInfo: Array<fileAccess.RootInfo> = rootInfos[0]; 86 let fileInfos: Array<fileAccess.FileInfo> = []; 87 let isDone: boolean = false; 88 let filter: Filter = {suffix : [".txt", ".jpg", ".xlsx"]}; // Set the filter. 89 try { 90 let fileIterator: string = rootInfo.listFile(); // Traverse the root directory of rootinfos[0] and return a FileIterator object. 91 // let fileIterator = rootInfo.scanFile(filter); // Filter device rootinfos[0] files that meet the specified conditions and return a FileIterator object. 92 if (!fileIterator) { 93 console.error("listFile interface returns an undefined object"); 94 } 95 while (!isDone) { 96 let result: boolean = fileIterator.next(); 97 console.info("next result = " + JSON.stringify(result)); 98 isDone = result.done; 99 if (!isDone) 100 fileInfos.push(result.value); 101 } 102 } catch (err) { 103 let error: BusinessError = err as BusinessError; 104 console.error("listFile failed, errCode:" + error.code + ", errMessage:" + error.message); 105 } 106 107 // Start from the specified directory. 108 let fileInfoDir: Array<fileAccess.FileInfo> = fileInfos[0]; // fileInfoDir indicates the information about a directory. 109 let subFileInfos: Array<fileAccess.FileInfo> = []; 110 let isDone02: boolean = false; 111 let filter02: Filter = {suffix : [".txt", ".jpg", ".xlsx"]}; // Set the filter. 112 try { 113 let fileIterator: string = fileInfoDir.listFile(); // Traverse files in the specified directory and return a FileIterator object. 114 // let fileIterator = rootInfo.scanFile(filter02); // Filter the files in the specified directory and return a FileIterator object. 115 if (!fileIterator) { 116 console.error("listFile interface returns an undefined object"); 117 } 118 while (!isDone02) { 119 let result: boolean = fileIterator.next(); 120 console.info("next result = " + JSON.stringify(result)); 121 isDone02 = result.done; 122 if (!isDone02) 123 subFileInfos.push(result.value); 124 } 125 } catch (err) { 126 let error: BusinessError = err as BusinessError; 127 console.error("listFile failed, errCode:" + error.code + ", errMessage:" + error.message); 128 } 129 ``` 130 1315. Perform operations on files or folders.<br> 132 You can integrate APIs of the user file access framework to implement user behaviors, such as deleting, renaming, creating, and moving a file or folder. The following example shows how to create a file. For details about other APIs, see [User File Access and Management](../reference/apis/js-apis-fileAccess.md). 133 134 ```ts 135 import { BusinessError } from '@ohos.base'; 136 137 // The local device is used as an example. 138 // Create a file. 139 // sourceUri is the URI in fileinfo of the Download directory. 140 // You need to use the obtained URI for development. 141 async function creatFile(): Promise<void> { 142 let sourceUri: string = "file://docs/storage/Users/currentUser/Download"; 143 let displayName: string = "file1"; 144 let fileUri: string = ''; 145 try { 146 // Obtain fileAccessHelperAllServer by referring to the sample code of fileAccess.createFileAccessHelper. 147 fileUri = await fileAccessHelperAllServer.createFile(sourceUri, displayName); 148 if (!fileUri) { 149 console.error("createFile return undefined object"); 150 } 151 console.info("createFile sucess, fileUri: " + JSON.stringify(fileUri)); 152 } catch (err) { 153 let error: BusinessError = err as BusinessError; 154 console.error("createFile failed, errCode:" + error.code + ", errMessage:" + error.message); 155 }; 156 } 157 ``` 158 159## Listening for Device Online/Offline Status 160 161For details about the APIs to be used, see [User File Access and Management](../reference/apis/js-apis-fileAccess.md). 162 163The **notify** interface can be used to observe not only the changes of directories, but also the device online/offline status. 164 165 1661. Apply for permissions required.<br> 167 168 Apply for the ohos.permission.FILE_ACCESS_MANAGER and ohos.permission.GET_BUNDLE_INFO_PRIVILEGED permissions. For details, see [Declaring Permissions](../security/AccessToken/declare-permissions.md). 169 170 > **NOTE** 171 > 172 > **ohos.permission.FILE_ACCESS_MANAGER** allows your application to use the user file access framework APIs. 173 > 174 > **ohos.permission.GET_BUNDLE_INFO_PRIVILEGED** allows your application to obtain information about file management server applications supported by the system. 175 1762. Import dependent modules. 177 178 ```ts 179 import fileAccess from '@ohos.file.fileAccess'; 180 import fileExtensionInfo from '@ohos.file.fileExtensionInfo'; 181 ``` 182 183 The **fileAccess** module provides APIs for basic file operations, and the **fileExtensionInfo** module provides key structs for application development. 184 1853. Define an observer callback. 186 187 ```ts 188 const callbackDir1 = (NotifyMessageDir: fileAccess.NotifyMessage) => { 189 if (NotifyMessageDir != undefined) { 190 console.log('NotifyType: ' + NotifyMessageDir.type + 'NotifyUri:' + NotifyMessageDir.uri[0]); 191 } else { 192 console.error("NotifyMessageDir is undefined"); 193 } 194 } 195 ``` 196 1974. Subscribe to the device online/offline status. 198 199 Pass in the constant [DEVICES_URI](../reference/apis/js-apis-fileAccess.md#) to the **registerObserver** method to listen for the device online/offline status. 200 201 ```ts 202 import { BusinessError } from '@ohos.base'; 203 async function UnregisterObserver03() { 204 try { 205 // Listen for the device online/offline status. 206 fileAccessHelper.registerObserver(fileAccess.DEVICES_URI, true, callbackDir1); 207 } catch (err) { 208 let error: BusinessError = err as BusinessError; 209 console.error("unregisterObserver failed, errCode:" + error.code + ", errMessage:" + error.message); 210 } 211 } 212 ``` 2135. Unsubscribe from the device online/offline status. 214 215 Pass in the constant [DEVICES_URI](../reference/apis/js-apis-fileAccess.md#) to the **unregisterObserver** method to unsubscribe from the device online/offline status. 216 217 ```ts 218 import { BusinessError } from '@ohos.base'; 219 try { 220 // Unsubscribe from the device online/offline status. 221 fileAccessHelper.unregisterObserver(fileAccess.DEVICES_URI, callbackDir1); 222 } catch (err) { 223 let error: BusinessError = err as BusinessError; 224 console.error("unregisterObserver failed, errCode:" + error.code + ", errMessage:" + error.message); 225 } 226 ``` 227