跳到主要内容

刷新本地图片和网络图片

在卡片上需要展示本地图片或从网络上下载的图片,获取本地图片和网络图片需要通过FormExtensionAbility来实现,如下示例代码介绍了如何在卡片上显示本地图片和网络图片。

  1. 下载网络图片需要使用到网络能力,需要申请ohos.permission.INTERNET权限,配置方式请参见声明权限

  2. 在WgtImgUpdateEntryFormAbility.ts文件中导入相关模块。

    // entry/src/main/ets/wgtimgupdateentryformability/WgtImgUpdateEntryFormAbility.ts
    import { Want } from '@kit.AbilityKit';
    import { BusinessError } from '@kit.BasicServicesKit';
    import { fileIo } from '@kit.CoreFileKit';
    import { formBindingData, FormExtensionAbility, formInfo, formProvider } from '@kit.FormKit';
    import { hilog } from '@kit.PerformanceAnalysisKit';
    import { http } from '@kit.NetworkKit';
  3. 在EntryFormAbility中的onAddForm生命周期回调中实现本地文件的刷新。

    // entry/src/main/ets/wgtimgupdateentryformability/WgtImgUpdateEntryFormAbility.ts
    const TAG: string = 'WgtImgUpdateEntryFormAbility';
    const DOMAIN_NUMBER: number = 0xFF00;
    // ...

    export default class WgtImgUpdateEntryFormAbility extends FormExtensionAbility {
    // 在添加卡片时,打开一个本地图片并将图片内容传递给卡片页面显示
    onAddForm(want: Want): formBindingData.FormBindingData {
    // 假设在当前卡片应用的tmp目录下有一个本地图片:head.PNG
    let tempDir = this.context.getApplicationContext().tempDir;
    hilog.info(DOMAIN_NUMBER, TAG, `tempDir: ${tempDir}`);
    let imgMap: Record<string, number> = {};
    try {
    // 打开本地图片并获取其打开后的fd, FormExtensionAbility进程销毁时释放
    let file = fileIo.openSync(tempDir + '/' + 'head.PNG');
    imgMap['imgBear'] = file.fd;
    } catch (e) {
    hilog.error(DOMAIN_NUMBER, TAG, `openSync failed: ${JSON.stringify(e as BusinessError)}`);
    }

    class FormDataClass {
    text: string = 'Image: Bear';
    loaded: boolean = true;
    // 卡片需要显示图片场景,必须和下列字段formImages中的key 'imgBear'相同。
    imgName: string = 'imgBear';
    // 卡片需要显示图片场景,必填字段(formImages不可缺省或改名), 'imgBear'对应fd
    formImages: Record<string, number> = imgMap;
    }

    let formData = new FormDataClass();
    // 将fd封装在formData中并返回至卡片页面
    return formBindingData.createFormBindingData(formData);
    }

    // ...
    }
  4. 在EntryFormAbility中的onFormEvent生命周期回调中实现网络文件的刷新。

    // entry/src/main/ets/wgtimgupdateentryformability/WgtImgUpdateEntryFormAbility.ts
    const TAG: string = 'WgtImgUpdateEntryFormAbility';
    const DOMAIN_NUMBER: number = 0xFF00;
    const TEXT1: string = '刷新中...'
    const TEXT2: string = '刷新失败'


    export default class WgtImgUpdateEntryFormAbility extends FormExtensionAbility {
    // ...
    async onFormEvent(formId: string, message: string): Promise<void> {
    let param: Record<string, string> = {
    'text': TEXT1
    };
    let formInfo: formBindingData.FormBindingData = formBindingData.createFormBindingData(param);
    formProvider.updateForm(formId, formInfo);

    // 注意:FormExtensionAbility在触发生命周期回调时被拉起,仅能在后台存在5秒
    // 建议下载能快速下载完成的小文件,如在5秒内未下载完成,则此次网络图片无法刷新至卡片页面上
    let netFile =
    'https://cn-assets.gitee.com/assets/mini_app-e5eee5a21c552b69ae6bf2cf87406b59.jpg';
    // 需要在此处使用真实的网络图片下载链接
    let tempDir = this.context.getApplicationContext().tempDir;
    let fileName = 'file' + Date.now();
    let tmpFile = tempDir + '/' + fileName;
    let imgMap: Record<string, number> = {};

    class FormDataClass {
    text: string = 'Image: Bear' + fileName;
    loaded: boolean = true;
    // 卡片需要显示图片场景,必须和下列字段formImages中的key fileName相同。
    imgName: string = fileName;
    // 卡片需要显示图片场景,必填字段(formImages不可缺省或改名), fileName对应fd
    formImages: Record<string, number> = imgMap;
    }

    let httpRequest = http.createHttp()
    let data = await httpRequest.request(netFile);
    if (data?.responseCode == http.ResponseCode.OK) {
    try {
    let imgFile = fileIo.openSync(tmpFile, fileIo.OpenMode.READ_WRITE | fileIo.OpenMode.CREATE);
    imgMap[fileName] = imgFile.fd;
    try {
    let writeLen: number = await fileIo.write(imgFile.fd, data.result as ArrayBuffer);
    hilog.info(DOMAIN_NUMBER, TAG, "write data to file succeed and size is:" + writeLen);
    hilog.info(DOMAIN_NUMBER, TAG, 'ArkTSCard download complete: %{public}s', tmpFile);
    try {
    let formData = new FormDataClass();
    let formInfo = formBindingData.createFormBindingData(formData);
    await formProvider.updateForm(formId, formInfo);
    hilog.info(DOMAIN_NUMBER, TAG, '%{public}s', 'FormAbility updateForm success.');
    } catch (error) {
    hilog.error(DOMAIN_NUMBER, TAG, `FormAbility updateForm failed: ${JSON.stringify(error)}`);
    }
    } catch (err) {
    hilog.error(DOMAIN_NUMBER, TAG,
    "write data to file failed with error message: " + err.message + ", error code: " + err.code);
    } finally {
    fileIo.closeSync(imgFile);
    }
    } catch (e) {
    hilog.error(DOMAIN_NUMBER, TAG, `openSync failed: ${JSON.stringify(e as BusinessError)}`);
    }

    } else {
    hilog.error(DOMAIN_NUMBER, TAG, `ArkTSCard download task failed`);
    let param: Record<string, string> = {
    'text': TEXT2
    };
    let formInfo: formBindingData.FormBindingData = formBindingData.createFormBindingData(param);
    formProvider.updateForm(formId, formInfo);
    }
    httpRequest.destroy();
    }

    onAcquireFormState(want: Want): formInfo.FormState {
    // 卡片使用方查询卡片状态时触发该回调,默认返回初始状态。
    return formInfo.FormState.READY;
    }

    }
  5. 在卡片页面通过backgroundImage属性展示EntryFormAbility传递过来的卡片内容。

    // entry/src/main/ets/widgetimageupdate/pages/WidgetImageUpdateCard.ets
    let storageWidgetImageUpdate = new LocalStorage();

    @Entry(storageWidgetImageUpdate)
    @Component
    struct WidgetImageUpdateCard {
    // $r('app.string.loading')需要替换为开发者所需的资源文件
    @LocalStorageProp('text') text: ResourceStr = $r('app.string.loading');
    @LocalStorageProp('loaded') loaded: boolean = false;
    // $r('app.string.imgName')需要替换为开发者所需的资源文件
    @LocalStorageProp('imgName') imgName: ResourceStr = $r('app.string.imgName');

    build() {
    Column() {
    Column() {
    Text(this.text)
    .fontColor('#FFFFFF')
    .opacity(0.9)
    .fontSize(12)
    .textOverflow({ overflow: TextOverflow.Ellipsis })
    .maxLines(1)
    .margin({ top: '8%', left: '10%' })
    }.width('100%').height('50%')
    .alignItems(HorizontalAlign.Start)

    Row() {
    Button() {
    // $r('app.string.update')需要替换为开发者所需的资源文件
    Text($r('app.string.update'))
    .fontColor('#45A6F4')
    .fontSize(12)
    }
    .width(120)
    .height(32)
    .margin({ top: '30%', bottom: '10%' })
    .backgroundColor('#FFFFFF')
    .borderRadius(16)
    .onClick(() => {
    postCardAction(this, {
    action: 'message',
    params: {
    info: 'refreshImage'
    }
    });
    })
    }.width('100%').height('40%')
    .justifyContent(FlexAlign.Center)
    }
    .width('100%').height('100%')
    // $r('app.media.ImageDisp')需要替换为开发者所需的资源文件
    .backgroundImage(this.loaded ? 'memory://' + this.imgName : $r('app.media.ImageDisp'))
    .backgroundImageSize(ImageSize.Cover)
    }
    }

  • Image组件入参格式为memory://fileName时表示进行远端内存图片显示,fileName来自EntryFormAbility传递对象('formImages': {key: fd})中的key。
  • Image组件通过传入的参数是否有变化来决定是否刷新图片,因此EntryFormAbility每次传递过来的imgName都需要不同,连续传递两个相同的imgName时,图片不会刷新。