1# 刷新本地图片和网络图片 2 3 4在卡片上通常需要展示本地图片或从网络上下载的图片,获取本地图片和网络图片需要通过FormExtensionAbility来实现,如下示例代码介绍了如何在卡片上显示本地图片和网络图片。 5 6 71. 下载网络图片需要使用到网络能力,需要申请ohos.permission.INTERNET权限,配置方式请参见[配置文件权限声明](../security/accesstoken-guidelines.md#配置文件权限声明)。 8 92. 在EntryFormAbility中的onAddForm生命周期回调中实现本地文件的刷新。 10 11 ```ts 12 import formBindingData from '@ohos.app.form.formBindingData'; 13 import FormExtensionAbility from '@ohos.app.form.FormExtensionAbility'; 14 import fs from '@ohos.file.fs'; 15 import Want from '@ohos.app.ability.Want'; 16 import Base from '@ohos.base'; 17 import fileFs from '@ohos.file.fs'; 18 19 export default class EntryFormAbility extends FormExtensionAbility { 20 ... 21 // 在添加卡片时,打开一个本地图片并将图片内容传递给卡片页面显示 22 onAddForm(want: Want) { 23 // 假设在当前卡片应用的tmp目录下有一个本地图片:head.PNG 24 let tempDir = this.context.getApplicationContext().tempDir; 25 // 打开本地图片并获取其打开后的fd 26 let file: fileFs.File; 27 let imgBear: Record<string, number>; 28 try { 29 file = fs.openSync(tempDir + '/' + 'head.PNG'); 30 imgBear = { 31 'imgBear': file.fd 32 } 33 } catch (e) { 34 console.error(`openSync failed: ${JSON.stringify(e as Base.BusinessError)}`); 35 } 36 class FormDataClass{ 37 text: string = 'Image: Bear' 38 imgName: string = 'imgBear' 39 loaded: boolean = true 40 formImages: Record<string, number> = imgBear 41 } 42 let formData = new FormDataClass(); 43 44 // 将fd封装在formData中并返回至卡片页面 45 return formBindingData.createFormBindingData(formData); 46 } 47 48 ... 49 } 50 ``` 51 523. 在EntryFormAbility中的onFormEvent生命周期回调中实现网络文件的刷新。 53 54 ```ts 55 import formBindingData from '@ohos.app.form.formBindingData'; 56 import formProvider from '@ohos.app.form.formProvider'; 57 import FormExtensionAbility from '@ohos.app.form.FormExtensionAbility'; 58 import request from '@ohos.request'; 59 import fs from '@ohos.file.fs'; 60 import Base from '@ohos.base'; 61 import fileFs from '@ohos.file.fs'; 62 63 export default class EntryFormAbility extends FormExtensionAbility { 64 // 在卡片页面触发message事件时,下载一个网络图片,并将网络图片内容传递给卡片页面显示 65 onFormEvent(formId: string, message: string) { 66 let param: Record<string, string> = { 67 'text': '刷新中...' 68 }; 69 let formInfo: formBindingData.FormBindingData = formBindingData.createFormBindingData(param); 70 formProvider.updateForm(formId, formInfo); 71 // 注意:FormExtensionAbility在触发生命周期回调时被拉起,仅能在后台存在5秒 72 // 建议下载能快速下载完成的小文件,如在5秒内未下载完成,则此次网络图片无法刷新至卡片页面上 73 let netFile = 'https://xxxx/xxxx.png'; // 需要在此处使用真实的网络图片下载链接 74 let tempDir = this.context.getApplicationContext().tempDir; 75 let fileName = 'file' + Date.now(); 76 let tmpFile = tempDir + '/' + fileName; 77 request.downloadFile(this.context, { 78 url: netFile, filePath: tmpFile, enableMetered: true, enableRoaming: true 79 }).then((task) => { 80 task.on('complete', () => { 81 console.info('ArkTSCard download complete:' + tmpFile); 82 let file: fileFs.File; 83 let fileInfo: Record<string, string | number> ={}; 84 try { 85 file = fs.openSync(tmpFile); 86 fileInfo[fileName] = file.fd; 87 } catch (e) { 88 console.error(`openSync failed: ${JSON.stringify(e as Base.BusinessError)}`); 89 } 90 class FormDataClass{ 91 text: string = 'Image: Bear' + fileName 92 imgName: string = fileName 93 loaded: boolean = true 94 formImages: object = fileInfo 95 } 96 let formData = new FormDataClass(); 97 98 let formInfo = formBindingData.createFormBindingData(formData); 99 formProvider.updateForm(formId, formInfo).then(() => { 100 console.info('FormAbility updateForm success.'); 101 }).catch((error: Base.BusinessError) => { 102 console.error('FormAbility updateForm failed: ' + JSON.stringify(error)); 103 }); 104 }) 105 task.on('fail', (err: number) => { 106 console.info('ArkTSCard download task failed. Cause:' + err); 107 let param: Record<string, string> = { 108 'text': '刷新失败' 109 }; 110 let formInfo: formBindingData.FormBindingData = formBindingData.createFormBindingData(param); 111 formProvider.updateForm(formId, formInfo); 112 }); 113 }).catch((err: Base.BusinessError) => { 114 console.error('Failed to request the download. Cause: ' + JSON.stringify(err)); 115 }); 116 } 117 }; 118 ``` 119 1204. 在卡片页面通过Image组件展示EntryFormAbility传递过来的卡片内容。 121 122 ```ts 123 let storage = new LocalStorage(); 124 @Entry(storage) 125 @Component 126 struct WidgetCard { 127 @LocalStorageProp('text') text: string = '加载中...'; 128 @LocalStorageProp('loaded') loaded: boolean = false; 129 @LocalStorageProp('imgName') imgName: string = 'name'; 130 131 build() { 132 Column() { 133 Text(this.text) 134 .fontSize('12vp') 135 .textAlign(TextAlign.Center) 136 .width('100%') 137 .height('20%') 138 139 Row() { 140 if (this.loaded) { 141 Image('memory://' + this.imgName) 142 .width('50%') 143 .height('50%') 144 .margin('5%') 145 } else { 146 Image('common/start.PNG') 147 .width('50%') 148 .height('50%') 149 .margin('1%') 150 } 151 }.alignItems(VerticalAlign.Center) 152 .justifyContent(FlexAlign.Center) 153 154 Button('刷新') 155 .height('15%') 156 .onClick(() => { 157 postCardAction(this, { 158 action: 'message', 159 params: { 160 info: 'refreshImage' 161 } 162 }); 163 }) 164 } 165 .width('100%').height('100%') 166 .alignItems(HorizontalAlign.Center) 167 .padding('5%') 168 } 169 } 170 ``` 171 172> **说明:** 173> - Image组件通过入参(**memory://fileName**)中的**memory://**标识来进行远端内存图片显示,其中**fileName**需要和EntryFormAbility传递对象(**'formImages': {key: fd})**中的**key**相对应。 174> 175> - Image组件通过传入的参数是否有变化来决定是否刷新图片,因此EntryFormAbility每次传递过来的**imgName**都需要不同,连续传递两个相同的**imgName**时,图片不会刷新。 176> 177> - 文件使用完成后必须关闭,否则会出现内存泄漏问题。系统不会自动关闭文件,开发者可以使用[fs.closeSync](../reference/apis/js-apis-file-fs.md#fsclosesync)来关闭文件。 178