【HarmonyOS】一文教你如何通过内存图片方式使用image组件加载网络图片资源

news/2024/7/21 11:13:02 标签: harmonyos, 华为

【关键字】

内存图片方式、image组件、网络图片资源、api6、服务卡片

1、写在前面

之前写过一篇元服务卡片的开发指导,有需求的可以参考以下文章:

【HarmonyOS】低代码开发之FA卡片开发流程

在2.6 初始化卡片部分,我们实现了加载网络资源的图片,但是直接使用image组件加载网络资源似乎在新版本设备上不太友好,所以今天我们来换一种实现方式。

2、代码实现

这次我们准备通过内存图片的方式来使用image组件加载网络图片资源,详细的教程可以参考官方文档:

通过内存图片方式使用image组件

我们的实现还是以【HarmonyOS】低代码开发之FA卡片开发流程 这篇文章中的代码为例,我们需要修改的是WidgetImpl.java这个类。

首先,我们要准备一个工具类,用户获取网络资源的字节流,这里我们定义HttpRequestUtils.java类,实现doGetRequestForFile()方法,返回一个字节数组,详情如下:

import ohos.hiviewdfx.HiLog;
import ohos.hiviewdfx.HiLogLabel;

import java.io.*;
import java.net.HttpURLConnection;
import java.net.URL;

public class HttpRequestUtils {
    private static final HiLogLabel TAG = new HiLogLabel(HiLog.DEBUG, 0x0, MainAbility.class.getName());

    public final static byte[] doGetRequestForFile(String urlStr) {
        InputStream is = null;
        HttpURLConnection conn = null;
        byte[] buff = new byte[1024];
        try {
            URL url = new URL(urlStr);
            conn = (HttpURLConnection) url.openConnection();

            conn.setDoInput(true);
            conn.setRequestMethod("GET");
            conn.setReadTimeout(6000);
            conn.connect();
            is = conn.getInputStream();
            if (conn.getResponseCode() == 200) {
                buff = readInputStream(is);
            } else{
                buff=null;
            }
        } catch (Exception e) {
            HiLog.error(TAG,"【获取图片异常】",e);
        }
        finally {
            try {
                if(is != null){
                    is.close();
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
            conn.disconnect();
        }
        return buff;
    }
}

然后我们进入WidgetImpl类中,我们需要修改初始化卡片数据以及点击事件时需要更新的卡片样式的代码,实现的思路都是一样的。

首先获取当前时间戳用来拼接图片名称,这样是为了保证唯一性:

long onlyOne = System.currentTimeMillis();

然后通过ZSONObject,将{imageSrc,memory:// + picName}的键值对添加到ZSONObject中:

object.put("btnSrc", "memory://btnsrc" + onlyOne + ".png");

然后使用该ZSONObject去创建一个FormBindingData:

FormBindingData bindingData = new FormBindingData(object);
ProviderFormInfo formInfo = new ProviderFormInfo();
formInfo.setJsBindingData(bindingData);

然后获取网络图片资源的字节流:

bytes = HttpRequestUtils.doGetRequestForFile(DataInstance.getInstance().getList().get(0).getBtnSrc());

最后通过addImageData接口添加数据:addImageData("xxx.png", bytes),其中,"xxx.png"为picName,必须和上面添加到ZSONObject中的键值对的picName一致,否则上面的内存图片路径("memory://xxx.png")将读取不到这里添加进去的图片数据。

bindingData.addImageData("btnsrc" + onlyOne + ".png", bytes);

WidgetImpl类的完整代码如下:

public class WidgetImpl extends FormController {

    private static final HiLogLabel TAG = new HiLogLabel(HiLog.DEBUG, 0x0, "jscard");
    private static final int DIMENSION_1X2 = 1;
    private static final int DIMENSION_2X2 = 2;
    private Context mContext;

    public WidgetImpl(Context context, String formName, Integer dimension) {
        super(context, formName, dimension);
        HiLog.info(TAG, "WidgetImpl");
        this.mContext = context;
    }

    @Override
    public ProviderFormInfo bindFormData(long formId) {
        HiLog.info(TAG, "222---" + new Date().getTime());
        HiLog.info(TAG, "bind form data");
        ZSONObject object = new ZSONObject();
        long onlyOne = System.currentTimeMillis();
        byte[] bytes = null;
        if (dimension == DIMENSION_1X2) {
            object.put("smallCard", true);
            object.put("normalCard", false);
            object.put("type", DataInstance.getInstance().getList().get(0).getType());
            object.put("title", DataInstance.getInstance().getList().get(0).getTitle());
            // 根据动态获取的云侧配置项里的字段值动态修改标题内容
            if (!TextUtils.isEmpty(DataInstance.getInstance().getExtFaParamStr())) {
                object.put("title", DataInstance.getInstance().getExtFaParamStr());
            }
        } else if (dimension == DIMENSION_2X2) {
            object.put("smallCard", false);
            object.put("normalCard", true);
            object.put("type", DataInstance.getInstance().getList().get(2).getType());
            object.put("title", DataInstance.getInstance().getList().get(2).getTitle());
            // 根据动态获取的云侧配置项里的字段值动态修改标题内容
            if (!TextUtils.isEmpty(DataInstance.getInstance().getExtFaParamStr())) {
                object.put("title", DataInstance.getInstance().getExtFaParamStr());
            }
        }
        object.put("btnSrc", "memory://btnsrc" + onlyOne + ".png");
        FormBindingData bindingData = new FormBindingData(object);
        ProviderFormInfo formInfo = new ProviderFormInfo();
        formInfo.setJsBindingData(bindingData);
        if (dimension == DIMENSION_1X2) {
            if (!DataInstance.getInstance().isPlay()) {
                bytes = HttpRequestUtils.doGetRequestForFile(DataInstance.getInstance().getList().get(0).getBtnSrc());
            } else {
                bytes = HttpRequestUtils.doGetRequestForFile(DataInstance.getInstance().getList().get(1).getBtnSrc());
            }
        } else if (dimension == DIMENSION_2X2) {
            if (!DataInstance.getInstance().isPlay()) {
                bytes = HttpRequestUtils.doGetRequestForFile(DataInstance.getInstance().getList().get(2).getBtnSrc());
            } else {
                bytes = HttpRequestUtils.doGetRequestForFile(DataInstance.getInstance().getList().get(3).getBtnSrc());
            }
        }
        bindingData.addImageData("btnsrc" + onlyOne + ".png", bytes);
        return formInfo;
    }

    @Override
    public void updateFormData(long formId, Object... vars) {
        HiLog.info(TAG, "updateFormData");
    }

    @Override
    public void onTriggerFormEvent(long formId, String message) {
        HiLog.info(TAG, "onTriggerFormEvent");
        if (!TextUtils.isEmpty(message)) {
            byte[] bytes = null;
            ZSONObject zsonObject = ZSONObject.stringToZSON(message);
            String msg = zsonObject.getString("message");
            if (!TextUtils.isEmpty(msg) && "play".equals(msg)) {
                if (!DataInstance.getInstance().isPlay()) {
                    DataInstance.getInstance().setPlay(true);
                    if (dimension == DIMENSION_1X2) {
                        bytes = HttpRequestUtils.doGetRequestForFile(DataInstance.getInstance().getList().get(1).getBtnSrc());
                    } else if (dimension == DIMENSION_2X2) {
                        bytes = HttpRequestUtils.doGetRequestForFile(DataInstance.getInstance().getList().get(3).getBtnSrc());
                    }
                    updateForm(formId, bytes);
                    play();
                } else {
                    DataInstance.getInstance().setPlay(false);
                    if (dimension == DIMENSION_1X2) {
                        bytes = HttpRequestUtils.doGetRequestForFile(DataInstance.getInstance().getList().get(0).getBtnSrc());
                    } else if (dimension == DIMENSION_2X2) {
                        bytes = HttpRequestUtils.doGetRequestForFile(DataInstance.getInstance().getList().get(2).getBtnSrc());
                    }
                    updateForm(formId, bytes);
                    pause();
                }
            }
        }
    }

    @Override
    public Class<? extends AbilitySlice> getRoutePageSlice(Intent intent) {
        return null;
    }

    // 更新数据
    private void updateForm(long formId, byte[] bb) {
        ZSONObject object = new ZSONObject();
        long onlyOne = System.currentTimeMillis();
        object.put("btnSrc", "memory://btnsrc" + onlyOne + ".png");
        FormBindingData bindingData = new FormBindingData(object);
        ProviderFormInfo formInfo = new ProviderFormInfo();
        formInfo.setJsBindingData(bindingData);
        bindingData.addImageData("btnsrc" + onlyOne + ".png", bb);
        try {
            if (mContext instanceof Ability) {
                ((Ability) mContext).updateForm(formId, bindingData);
            }
        } catch (FormException e) {
            HiLog.error(TAG, e.getMessage());
        }
    }

    // 暂停音乐
    private void pause() {
        DataInstance.getInstance().getPlayer().pause();
    }

    // 播放音乐
    private void play() {
        DataInstance.getInstance().getPlayer().play();
    }
}

OK,通过上面的代码,我们就可以实现使用image组件加载网络图片资源啦。


http://www.niftyadmin.cn/n/5040374.html

相关文章

研究报告:周界警戒AI算法+视频智能分析在安全生产场景中的应用

长期以来&#xff0c;周界防范安防系统在大型园区、工厂、社区、机场、火车站站台、重点单位等领域应用较为广泛和常见。随着AI人工智能等新兴技术的快速发展与落地应用&#xff0c;通过AI智能检测与视频智能分析技术&#xff0c;现代化的周界安防系统可以做到全天候快速、准确…

Qt/C++音视频开发54-视频监控控件的极致设计

一、前言 跌跌撞撞摸爬滚打一步步迭代完善到今天&#xff0c;这个视频监控控件的设计&#xff0c;在现阶段水平上个人认为是做的最棒的&#xff08;稍微自恋一下&#xff09;&#xff0c;理论上来说应该可以用5年不用推翻重写&#xff0c;推翻重写当然也是程序员爱干的事情&am…

Visual Studio Code从GIT拉取vue项目并运行

安装Visual Studio Code 安装GIT 安装node.js&#xff0c;配置好环境变量 拉取项目 文章一 文章二 运行项目 文章一 提交代码 文章一

Golang slice 通过growslice调用nextslicecap计算扩容

先来看一段代码 code: e : []int64{1, 2, 3}fmt.Println("cap of e before:", cap(e))e append(e, 4, 5, 6, 7)fmt.Println("cap of e after:", cap(e))output:cap of e before: 3 cap of e after: 8 为什么容量是8&#xff1f; append了的4个元素&…

MySQL只同步单个表或多个表,非全部同步!

replicate-do-table 是 MySQL 复制配置中的一个选项&#xff0c;它允许您指定要在从服务器上复制的表。如果您想要只复制主服务器上特定的表到从服务器&#xff0c;您可以使用这个选项。 以下是如何操作 replicate-do-table 的步骤&#xff1a; 停止从服务器: 在从服务器上执行…

Kafka 笔记 (Non-Root/Container)

目录 1. Kafka 笔记 (Non-Root/Container)1.1. 启动1.2. bitnami/kafka1.2.1. Non-Root Containers 1. Kafka 笔记 (Non-Root/Container) 1.1. 启动 Kafka 需要与 ZooKeeper 一起启动: Kafka with ZooKeeper Run the following commands in order to start all services in…

Python经典小游戏pygame:五子棋

pygame是一个用于制作2D游戏的Python库。它提供了许多用于处理图像、声音、事件和碰撞检测等游戏元素的函数和方法。 #我的Python教程 #官方微信公众号&#xff1a;wdPython五子棋起源于中国&#xff0c;是全国智力运动会竞技项目之一&#xff0c;是一种两人对弈的纯策略型棋类…

【算法】矩阵快速幂优化动态规划

文章目录 知识讲解题目列表[矩阵快速幂] 题目列表&#x1f4d5;70. 爬楼梯解法1——线性DP解法2——矩阵快速幂 509. 斐波那契数1137. 第 N 个泰波那契数1220. 统计元音字母序列的数目解法1——线性DP解法2——矩阵快速幂优化DP 552. 学生出勤记录 II&#xff08;&#x1f6b9;…