OpenHarmony鸿蒙南向开发案例:【智能门铃】

样例简介

智能门铃通过监控来访者信息,告诉主人门外是否有人按铃、有陌生人靠近或者无人状态。主人可以在数字管家中远程接收消息,并根据需要进行远程取消报警和一键开锁。同时,也可以通过室内屏幕获取门外状态。室内屏幕显示界面使用DevEco Studio 编写的js应用,具有很好的兼容和移植特性。硬件上采用了带有HDF框架的驱动模型,通过GPIO来获取传感器采集信息并驱动报警器进行动作。

image-20211228172423071

运行效果

image-20220402103912059

样例原理

如上图所示,智能门铃整体方案原理图可以大致分成:智能门铃设备、数字管家应用、云平台三部分。智能门铃通过MQTT协议连接华为IOT物联网平台,从而实现命令的接收和属性上报。 。

工程版本
  • 系统版本/API版本:OpenHarmony 3.1 Beta
  • hb版本:0.4.4
  • 工具链版本:gcc-arm-none-eabi-10.3-2021.10
  • 鸿蒙开发指导文档:gitee.com/li-shizhen-skin/harmony-os/blob/master/README.md点击或者复制转到。

快速上手

准备硬件环境
  • 欧智通V200Z-R开发板
  • 润和智能家居套件-红绿灯板
  • 润和智能家居套件-炫彩灯板
  • 预装HarmonyOS手机一台

    注:HarmonyOS是华为基于开源项目OpenHarmony开发的面向多种全场景智能设备的商用版本

准备开发环境

HarmonyOS与OpenHarmony鸿蒙文档籽料:mau123789是v直接拿

搜狗高速浏览器截图20240326151547.png

安装必备软件

开发基础环境由 windows 工作台和 Linux 编译服务器组成。windows 工作台可以通过 samba 服务或 ssh 方式访问 Linux编译服务器。其中 windows 工作台用来烧录和代码编辑,Linux 编译服务器用来编译 OpenHarmony 代码,为了简化步骤,Linux 编译服务器推荐安装 Ubuntu20.04。

安装编译依赖基础软件
sudo apt-get install -y build-essential gcc g++ make zlib* libffi-dev git git-lfs
安装和配置Python
  • 打开Linux终端。
  • 输入如下命令,查看python版本号,需要使用python3.7以上版本。
python3 --version
  • 安装并升级Python包管理工具(pip3)。
sudo apt-get install python3-setuptools python3-pip -y
sudo pip3 install --upgrade pip
  • 设置pip的国内镜像
pip3 install -i https://pypi.tuna.tsinghua.edu.cn/simple requests
安装hb
  • 输入如下命令确认hb是否为version 0.4.4
hb -v

a. 若提示如下内容,则表示未安装可以从第2步开始操作。

bash: /home/***/.local/bin/hb: No such file or directory

b.若提示如下内容,需要先卸载该版本,然后再执行第2步操作步骤。

[OHOS INFO] hb version 0.4.3

卸载命令:

pip3 uninstall ohos-build
  • 运行如下命令安装hb
pip3 install build/lite   // 该命令需在OpenHarmony源码根目录下执行
  • 设置环境变量
vim ~/.bashrc

将以下命令拷贝到.bashrc文件的最后一行,保存并退出。

export PATH=~/.local/bin:$PATH

执行如下命令更新环境变量。

source ~/.bashrc
  • 再次执行”hb -v“,有以下版本显示则表示安装的hb版本正确。
[OHOS INFO] hb version 0.4.4
安装交叉编译环境

在Linux编译服务器上搭建好基础开发环境后,需要安装OpenHarmony 编译V200Z-R平台特有的开发环境。

安装arm-none-eabi-gcc
  • 打开Linux终端。
  • 下载[arm-none-eabi-gcc]编译工具。
  • 安装[arm-none-eabi-gcc]

解压 [gcc-arm-none-eabi-10.3-2021.10-x86_64-linux.tar.bz2]安装包至~/toolchain/路径下。

mkdir -p ~/toolchain/
tar -jxvf gcc-arm-none-eabi-10.3-2021.10-x86_64-linux.tar.bz2 -C ~/toolchain/

设置环境变量。

vim ~/.bashrc

将以下命令拷贝到.bashrc文件的最后一行,保存并退出。

export PATH=~/toolchain/gcc-arm-none-eabi-10.3-2021.10/bin:$PATH

生效环境变量。

source ~/.bashrc
  • 在命令行中输入如下命令,如果能正确显示编译器版本号,表明编译器安装成功。
arm-none-eabi-gcc -v
准备工程

本用例采用repo的方式从码云官仓下载系统系统源码以及开发板适配代码,使用git从gitee的sig仓库拉取设备应用代码。

配置git
  • 提前注册准备码云gitee账号。
  • git工具下载安装

    sudo apt install git
    sudo apt install git-lfs
  • 生成/添加SSH密钥:生成密钥

    使用gitee账号绑定的邮箱生成密钥对

    ssh-keygen -t ed25519 -C "xxxxx@xxxxx.com"
  • 查看生成的密钥

    cat ~/.ssh/id_ed25519.pub
  • 复制生成后的 ssh key,返回gitee个人主页,通过主页 「个人设置」->「安全设置」->「SSH 公钥」 ,将生成的“SSH密钥”添加到仓库中。
  • 配置git用户信息

    git config --global user.name "yourname"
    git config --global user.email "your-email-address"
    git config --global credential.helper store
准备repo
sudo curl https://gitee.com/oschina/repo/raw/fork_flow/repo-py3 > /usr/local/bin/repo
chmod a+x /usr/local/bin/repo
pip3 install -i https://pypi.tuna.tsinghua.edu.cn/simple requests
准备系统源码
#特别注意:请下载OpenHarmony 3.1 Beta版本
mkdir ~/OpenHarmony_3.1_Beta
cd ~/OpenHarmony_3.1_Beta
repo init -u git@gitee.com:openharmony/manifest.git -b refs/tags/OpenHarmony-v3.1-Beta --no-repo-verify
repo sync -c
repo forall -c 'git lfs pull'

注意:

  • 权限问题请参考[生成/添加SSH公钥]。
  • 若在已安装python3.8后,执行repo init 时,仍显示如下错误:
/usr/bin/env: ‘python’: No such file or directory

执行如下命令后,进行重试:

sudo ln -s /usr/bin/python3.8 /usr/bin/python
准备设备侧应用代码

使用git 命令下载

git clone git@gitee.com:openharmony-sig/knowledge_demo_smart_home.git --depth=1
编译前准备
  • 代码拷贝

team_x源码拷贝

mkdir ~/OpenHarmony_3.1_Beta/vendor/team_x
cp -rfa ~/knowledge_demo_smart_home/dev/team_x/smart_door_bell ~/OpenHarmony_3.1_Beta/vendor/team_x

common库拷贝

cp -rfa ~/knowledge_demo_smart_home/dev/team_x/common ~/OpenHarmony_3.1_Beta/vendor/team_x

iot_link库拷贝

cp -rfa ~/knowledge_demo_smart_home/dev/third_party/iot_link ~/OpenHarmony_3.1_Beta/third_party/
  • kernel/liteos_m 修改

步骤1、下载patch

​ 点击上述链接进入浏览器,将该网页中内容全部复制。

​ 本地创建一个名为bes_kernle.patch文件,并将已经复制的内容粘贴到该文件中。

步骤2、 打上步骤1中的patch

cd ~/OpenHarmony_3.1_Beta/kernel/liteos_m
patch -p1 < bes_kernle.patch
  • device/soc/bestechnic 修改

步骤1、下载patch

​ 点击上述链接进入浏览器,将该网页中内容全部复制。

​ 本地创建一个名为bes_device.patch文件,并将已经复制的内容粘贴到该文件中。

步骤2、 打上步骤1中的patch

cd ~/OpenHarmony_3.1_Beta/device/soc/bestechnic
patch -p1 < bes_device.patch
  • third_party/mbedtls 修改

参考如下代码段修改 mbedtls/library/platform.c

diff --git a/library/platform.c b/library/platform.c
index c4c3fd3..214173b 100755
--- a/library/platform.c
+++ b/library/platform.c
@@ -86,9 +86,24 @@ static void platform_free_uninit( void *ptr )static void * (*mbedtls_calloc_func)( size_t, size_t ) = MBEDTLS_PLATFORM_STD_CALLOC;static void (*mbedtls_free_func)( void * ) = MBEDTLS_PLATFORM_STD_FREE;+#include "los_memory.h"
+void * mbedtls_calloc( size_t nmemb, size_t size ){
-    return (*mbedtls_calloc_func)( nmemb, size );
+    //return (*mbedtls_calloc_func)( nmemb, size );
+    size_t real_size;
+    void *ptr = NULL;
+
+    if (nmemb == 0 || size == 0) {
+        return NULL;
+    }
+
+    real_size = (size_t)(nmemb * size);
+    ptr = LOS_MemAlloc(OS_SYS_MEM_ADDR, real_size);
+    if (ptr != NULL) {
+        (void)memset_s(ptr, real_size, 0, real_size);
+    }
+    return ptr;}

参考如下代码段修改 mbedtls/BUILD.gn

diff --git a/BUILD.gn b/BUILD.gn
index 9ecb37a..30dbb2e 100755
--- a/BUILD.gn
+++ b/BUILD.gn
@@ -124,6 +124,7 @@ if (ohos_kernel_type == "liteos_m") {include_dirs = ["//kernel/liteos_m/kal/posix/include","//kernel/liteos_m/kernel/include","//kernel/liteos_m/utils","//third_party/musl/porting/liteos_m/kernel/include/",
+        "//kernel/liteos_m/kernel/include/",]}output_name = "mbedtls"
  • third_party/lwip 修改

修改src/api/netdb.c 文件

diff --git a/src/api/netdb.c b/src/api/netdb.c
index 52a6fdf..2043636 100644
--- a/src/api/netdb.c
+++ b/src/api/netdb.c
@@ -100,7 +100,7 @@ lwip_gethostbyname(const char *name)err = netconn_gethostbyname(name, &addr);if (err != ERR_OK) {LWIP_DEBUGF(DNS_DEBUG, ("lwip_gethostbyname(%s) failed,   err=%d\n", name, err));
-    h_errno = HOST_NOT_FOUND;
+    //h_errno = HOST_NOT_FOUND;return NULL;}
  • third_party/iot_link 修改

参考如下代码段修改 third_party/iot_link/network/dtls/BUILD.gn

diff --git a/third_party/iot_link/network/dtls/BUILD.gn b/third_party/iot_link/network/dtls/BUILD.gn
index 035805d709075ef2bc9d9388e73edb84dc1ee943..ac7256cf2bc93a0a998388f31910f152bf306fa9 100755
--- a/third_party/iot_link/network/dtls/BUILD.gn
+++ b/third_party/iot_link/network/dtls/BUILD.gn
@@ -41,7 +41,8 @@ dtls_cflags = ["-Wno-unused-parameter",]-static_library("dtls") {
+#static_library("dtls") {
+source_set("dtls") {cflags = dtls_cflagsdefines = dtls_defsources = dtls_src

参考如下代码段修改 third_party/iot_link/network/dtls/mbedtls/mbedtls_port/dtls_interface.c

diff --git a/third_party/iot_link/network/dtls/mbedtls/mbedtls_port/dtls_interface.c b/third_party/iot_link/network/dtls/mbedtls/mbedtls_port/dtls_interface.c
index 9f87c6fae041b17af4522d5d89114b69005a00be..73e406893a33385a9888a82cf6d1b3706a81387d 100755
--- a/third_party/iot_link/network/dtls/mbedtls/mbedtls_port/dtls_interface.c
+++ b/third_party/iot_link/network/dtls/mbedtls/mbedtls_port/dtls_interface.c
@@ -56,6 +56,7 @@#include "dtls_interface.h"#include "mbedtls/net_sockets.h"#include "mbedtls/ssl_internal.h"
+#include "mbedtls/platform.h"#include "cmsis_os2.h"#include "string.h"@@ -157,6 +158,7 @@ mbedtls_ssl_context *dtls_ssl_new(dtls_establish_info_s *info, char plat_type)#if defined(MBEDTLS_KEY_EXCHANGE__SOME__PSK_ENABLED)if (info->psk_or_cert == VERIFY_WITH_PSK){
+      #if 0if ((ret = mbedtls_ssl_conf_psk(conf,info->v.p.psk,info->v.p.psk_len,
@@ -166,6 +168,7 @@ mbedtls_ssl_context *dtls_ssl_new(dtls_establish_info_s *info, char plat_type)MBEDTLS_LOG("mbedtls_ssl_conf_psk failed: -0x%x", -ret);goto exit_fail;}
+        #endif}#endif@@ -548,5 +551,5 @@ void dtls_init(void){(void)mbedtls_platform_set_calloc_free(calloc, free);(void)mbedtls_platform_set_snprintf(snprintf);
-    (void)mbedtls_platform_set_printf(printf);
+    //(void)mbedtls_platform_set_printf(printf);}

参考如下代码段修改 third_party/iot_link/network/mqtt/BUILD.gn

diff --git a/third_party/iot_link/network/mqtt/BUILD.gn b/third_party/iot_link/network/mqtt/BUILD.gn
index 5a4a8e0ded129b91ce550a5d16f04e0784fddd45..d6bc39b8ede5735d79c74a6ab562c6141bd0b377 100755
--- a/third_party/iot_link/network/mqtt/BUILD.gn
+++ b/third_party/iot_link/network/mqtt/BUILD.gn
@@ -16,6 +16,7 @@ mqtt_paho_inc = ["paho_mqtt/paho/MQTTPacket/src","paho_mqtt/port","../../inc",
+    "//third_party/musl/porting/liteos_m/kernel/include/","//kernel/liteos_m/components/cmsis/2.0","//vendor/hisi/hi3861/hi3861/third_party/lwip_sack/include/",]
@@ -50,7 +51,8 @@ mqtt_cflags = ["-Wno-unused-function",]-static_library("mqtt") {
+#static_library("mqtt") {
+source_set("mqtt") {cflags = mqtt_cflagsdefines = mqtt_paho_defssources = mqtt_paho_src

参考如下代码段修改 third_party/iot_link/network/mqtt/paho_mqtt/port/paho_mqtt_port.c

diff --git a/third_party/iot_link/network/mqtt/paho_mqtt/port/paho_mqtt_port.c b/third_party/iot_link/network/mqtt/paho_mqtt/port/paho_mqtt_port.c
index a213b74c7286c68061e3285c0c3ac6c440949f26..d0900c8d88149f08c93fe885f34f4d2affa30885 100755
--- a/third_party/iot_link/network/mqtt/paho_mqtt/port/paho_mqtt_port.c
+++ b/third_party/iot_link/network/mqtt/paho_mqtt/port/paho_mqtt_port.c
@@ -180,6 +180,50 @@ static int __socket_read(void *ctx, unsigned char *buf, int len, int timeout)int fd;int ret = 0;+#if 1
+        int bytes = 0;
+    fd_set fdset;
+
+    struct timeval timedelay = {timeout / 1000, (timeout % 1000) * 1000};
+    if(NULL== buf)
+    {
+        return ret;
+    }
+
+    fd = (int)(intptr_t)ctx;  ///< socket could be zero
+
+    if (timedelay.tv_sec < 0 || (timedelay.tv_sec == 0 && timedelay.tv_usec <= 0))
+    {
+        timedelay.tv_sec = 0;
+        timedelay.tv_usec = 100;
+    }
+    timedelay.tv_sec = 5;
+    FD_ZERO(&fdset);
+    FD_SET(fd, &fdset);
+
+    ret = select(fd + 1, &fdset, NULL, NULL, &timedelay);
+    printf("[%s|%s|%d]fd = %d, ret = %d\n", __FILE__,__func__,__LINE__, fd, ret);
+    if (ret > 0) {
+        while (bytes < len) {
+            int rc = recv(fd, &buf[bytes], (size_t)(len - bytes), 0);
+            printf("[%s|%s|%d]fd = %d, rc = %d\n", __FILE__,__func__,__LINE__, fd, rc);
+            if (rc == -1) {
+                if (errno != EAGAIN && errno != EWOULDBLOCK) {
+                    bytes = -1;
+                }
+                break;
+            } else if (rc == 0) {
+                bytes = 0;
+                break;
+            } else {
+                bytes += rc;
+            }
+        }
+   }
+
+   return bytes;
+#else
+struct timeval timedelay = {timeout / 1000, (timeout % 1000) * 1000};if(NULL== buf){
@@ -216,7 +260,7 @@ static int __socket_read(void *ctx, unsigned char *buf, int len, int timeout)}}return bytes;
-
+  #endif}@@ -586,7 +630,7 @@ static void * __connect(mqtt_al_conpara_t *conparam)attr.cb_mem = NULL;attr.cb_size = 0U;attr.stack_mem = NULL;
-    attr.stack_size = 1024 * 2;
+    attr.stack_size = 1024 * 4;attr.priority = 28;cb->task = osThreadNew((osThreadFunc_t)__loop_entry, cb, &attr);if(NULL == cb->task)

参考如下代码段修改 third_party/iot_link/network/mqtt/paho_mqtt/port/paho_osdepends.h

diff --git a/third_party/iot_link/network/mqtt/paho_mqtt/port/paho_osdepends.h b/third_party/iot_link/network/mqtt/paho_mqtt/port/paho_osdepends.h
index 2c6cab1b90bef7be82891637c200cdb664e9735b..38e3dce3867d2b2d538858c72b36bee7330c7b4f 100755
--- a/third_party/iot_link/network/mqtt/paho_mqtt/port/paho_osdepends.h
+++ b/third_party/iot_link/network/mqtt/paho_mqtt/port/paho_osdepends.h
@@ -84,6 +84,7 @@#include "ohos_init.h"#include "cmsis_os2.h"
+#include "sys/time.h"#include  <mqtt_al.h>#define MQTT_TASK 1
@@ -117,6 +118,13 @@ typedef struct Threadint ThreadStart(Thread*, void (*fn)(void*), void* arg);+#define timeradd(s,t,a) (void) ( (a)->tv_sec = (s)->tv_sec + (t)->tv_sec, \
+        ((a)->tv_usec = (s)->tv_usec + (t)->tv_usec) >= 1000000 && \
+        ((a)->tv_usec -= 1000000, (a)->tv_sec++) )
+#define timersub(s,t,a) (void) ( (a)->tv_sec = (s)->tv_sec - (t)->tv_sec, \
+        ((a)->tv_usec = (s)->tv_usec - (t)->tv_usec) < 0 && \
+        ((a)->tv_usec += 1000000, (a)->tv_sec--) )
+typedef struct Network{void *ctx;                      ///< if it is tls, then it is tls context, else it is socket fd

参考如下代码段修改 third_party/iot_link/oc_mqtt/BUILD.gn

diff --git a/third_party/iot_link/oc_mqtt/BUILD.gn b/third_party/iot_link/oc_mqtt/BUILD.gn
index c777c5e4f2e79646a5c76471f22c5437e9d4b3ee..1e3e07ddb6c0bc52465a60c415c8550608a21e55 100755
--- a/third_party/iot_link/oc_mqtt/BUILD.gn
+++ b/third_party/iot_link/oc_mqtt/BUILD.gn
@@ -11,7 +11,8 @@# See the License for the specific language governing permissions and# limitations under the License.-static_library("oc_mqtt") {
+#static_library("oc_mqtt") {
+source_set("oc_mqtt") {sources = ["oc_mqtt_al/oc_mqtt_al.c","oc_mqtt_profile_v5/oc_mqtt_profile.c",
  • communicationkit相关修改

参考下方代码修改foundation/ace/ace_engine_lite/frameworks/module_manager/ohos_module_config.h

diff --git a/frameworks/module_manager/ohos_module_config.h b/frameworks/module_manager/ohos_module_config.h
index f8eb744..3bb25d1 100644
--- a/frameworks/module_manager/ohos_module_config.h
+++ b/frameworks/module_manager/ohos_module_config.h
@@ -47,6 +47,10 @@ extern void InitNativeApiFs(JSIValue exports);extern void InitNativeApiKv(JSIValue exports);#endif+#if (FEATURE_MODULE_COMMUNICATIONKIT == 1)
+extern void InitNativeApiCommunicationKit(JSIValue exports);
+#endif
+#if (FEATURE_MODULE_DEVICE == 1)extern void InitDeviceModule(JSIValue exports);#endif
@@ -98,11 +102,13 @@ const Module OHOS_MODULES[] = {#if (FEATURE_MODULE_DIALOG == 1){"prompt", InitDialogModule},#endif // FEATURE_MODULE_DIALOG
-#if (FEATURE_MODULE_STORAGE == 1){"file", InitNativeApiFs},{"storage", InitNativeApiKv},#endif
+#if (FEATURE_MODULE_COMMUNICATIONKIT == 1)
+    {"CommunicationKit", InitNativeApiCommunicationKit},
+#endif#if (FEATURE_MODULE_DEVICE == 1){"device", InitDeviceModule},#endif
  • flash_size配置修改

根据开发板的硬件版本不同,还需更改配置文件中flash_size的值。具体路径为/device/board/fnlink/v200zr/liteos_m/config.gni。其中v1.0版本对应16,2.0版本对应32,硬件版本打印在了开发板正面。

diff --git a/v200zr/liteos_m/config.gni b/v200zr/liteos_m/config.gni
index b288101..6489dd2 100644
--- a/v200zr/liteos_m/config.gni
+++ b/v200zr/liteos_m/config.gni
@@ -47,7 +47,7 @@ if (product_path != "") {bsp_bin_list = product_conf.bsp_bin_listpack_burn = product_conf.pack_burn}
-flash_size = 16
+flash_size = 32
  • hdf配置文件修改

参考下方代码修改device/board/fnlink/shields/v200zr-evb-t1/v200zr-evb-t1.hcs

diff --git a/shields/v200zr-evb-t1/v200zr-evb-t1.hcs b/shields/v200zr-evb-t1/v200zr-evb-t1.hcs
index 44212eb..4fc99da 100644
--- a/shields/v200zr-evb-t1/v200zr-evb-t1.hcs
+++ b/shields/v200zr-evb-t1/v200zr-evb-t1.hcs
@@ -17,12 +17,17 @@ root {platform {gpio_config {match_attr = "gpio_config";
-            pin = [0, 1];
+            pin = [0, 1, 2, 3, 4, 5, 6, 7];// touch_ztw523: TSP_RST - GPIO12, TSP_INT-GPIO27// touch_fts: TSP_RST - GPIO05, TSP_INT-GPIO27
-            realPin = [5, 27];
-            config = [5, 2];
-            pinNum = 2;
+            realPin = [5, 27, 1, 11, 15, 10, 14, 13];
+            config = [5, 2, 2, 4, 5, 5, 5, 2];
+            pinNum = 8;
+        }
+        pwm0_config{
+            match_attr = "pwm0_config";
+            pwmId = 0;
+            pwmPin = 20;}i2c_config {i2c0 {
  • gpio hdf框架相关修改。

根据下方代码修改drivers/adapter/platform/gpio/gpio_bes.c

diff --git a/platform/gpio/gpio_bes.c b/platform/gpio/gpio_bes.c
index ed4d18b..890d528 100755
--- a/platform/gpio/gpio_bes.c
+++ b/platform/gpio/gpio_bes.c
@@ -257,7 +257,7 @@ static int32_t GpioDriverBind(struct HdfDeviceObject *device)}gpioCntlr.device.hdfDev = device;
-    device->service = gpioCntlr.device.service;
+    device->service = &(gpioCntlr.device);return HDF_SUCCESS;}
  • 将pwm_if.c纳入编译。

参考下方代码修改drivers\adapter\khdf\liteos_m\platform\BUILD.gn

diff --git a/khdf/liteos_m/platform/BUILD.gn b/khdf/liteos_m/platform/BUILD.gn
index 84c9152..8a4016a 100755
--- a/khdf/liteos_m/platform/BUILD.gn
+++ b/khdf/liteos_m/platform/BUILD.gn
@@ -63,7 +63,10 @@ hdf_driver("hdf_platform_lite") {}if (defined(LOSCFG_DRIVERS_HDF_PLATFORM_PWM)) {
-    sources += [ "$HDF_FRAMEWORKS_PATH/support/platform/src/pwm/pwm_core.c" ]
+    sources += [
+      "$HDF_FRAMEWORKS_PATH/support/platform/src/pwm/pwm_core.c",
+      "$HDF_FRAMEWORKS_PATH/support/platform/src/pwm/pwm_if.c",
+    ]}if (defined(LOSCFG_DRIVERS_HDF_PLATFORM_RTC)) {
  • 将JS应用合入工程。

关于JS应用开发请参考从零开始学习L0: JS开发系列。本节为可选章节,忽略本节内容不影响智能门铃项目的展示和运行。但若想要在本项目的基础上进行涉及到显示的内容修改与新规开发,例如增加天气预报的图标,则需完整的阅读本节及链接内容。

  • 下载并安装[DevEco Studio]。

  • 打开js工程。

工程目录为:knowledge_demo_smart_home/dev/team_x/smart_door_bell/FA

  • 在DevEco Studio的SDK中添加@system.communicationkit.d.ts文件。

将@system.communicationkit.d.ts文件(源文件目录:~/knowledge_demo_smart_home/dev/interface/sdk-js/api/common/@system.communicationkit.d.ts)拷贝到在DevEco Studio的SDK中(目标目录:HarmonyOS Legacy SDK/js/3.0.0.0/api/common)。HarmonyOS Legacy SDK目录在DevEco Studio安装时,由用户配置,该目录位置可在设置(ctrl+alt+s)中查找。

  • 编译hap包。

依次选择构建 -> Build Hap(s)/APP(s) -> Build Hap(s)进行hap包编译,编译后的hap包目录为:FA\entry\build\outputs\hap\debug\entry-debug-lite-unsigned.hap。

  • 将js代码合入OpenHarmonySDK。

将entry-debug-lite-unsigned.hap修改后缀为zip并解压。在解压后的目录:entry-debug-lite-unsigned/assets/js/default目录中除app.js.map外的的数据全部拷贝到OpenHarmonySDK中的vendor/team_x/smart_door_bell/fs/data/data/js目录下。

工程效果

整合并修改完成后的代码目录结构如下图:

image-20211228205514766

编译

编译命令:

hb set  // 如果是第一次编译,Input code path 命令行中键入"./" 指定OpenHarmony工程编译根目录后回车。

image-20211228205557170

如下图所示,使用键盘上下键选中智能门铃 “smart_door_bell”(注:工程名字根据实际要编译的工程来):

hb build // 如果需要全量编译,可以添加-f 选项

编译通过,生成固件成功,如图:

image-20211228205659402

烧录/安装
  • 安装[CP2102驱动]
  • 固件编译完成以后拷贝/out/v200zr/smart_door_bell/write_flash_gui文件夹到windows下,并点击Wifi_download_main.exe

  • 点击工具上的文件夹图标

  • 选择List按钮

  • 在显示出来的串口列表中选择需要烧录的串口,并点击开始按钮。

  • 在开发板上点击reset按键,或者重新上电。

  • 进入烧录状态

  • 烧录成功

​ 注意:固件烧录完成后需将烧录窗口关闭,再按下设备Reset键设备才能启动。如果未将烧录窗口关闭,设备会再次进行烧写状态。

操作体验
设备配网
  • 在设备上电前需准备好安装了数字管家应用的HarmonyOS手机, 并在设置中开启手机的NFC功能;
  • 写设备NFC标签;
  • 烧录完成后,上电。开发者在观察开发板上状态LED灯以1Hz的频率闪烁时,将手机上半部靠近开发板NFC标签处;
  • 手机将自动拉起数字管家应用并进入配网状态,根据手机提示连接到设备热点teamX-DoorBell01后返回数字管家APP,输入路由账号密码,点击确定后,手机将信息传输到设备侧,设备开始联网。

设备控制
  • 有陌生人靠近:

  • 有人按门铃:

  • 远程一键开门:

鸿蒙开发岗位需要掌握那些核心要领?

目前还有很多小伙伴不知道要学习哪些鸿蒙技术?不知道重点掌握哪些?为了避免学习时频繁踩坑,最终浪费大量时间的。

自己学习时必须要有一份实用的鸿蒙(Harmony NEXT)资料非常有必要。 这里我推荐,根据鸿蒙开发官网梳理与华为内部人员的分享总结出的开发文档。内容包含了:【ArkTS、ArkUI、Stage模型、多端部署、分布式应用开发、音频、视频、WebGL、OpenHarmony多媒体技术、Napi组件、OpenHarmony内核、Harmony南向开发、鸿蒙项目实战】等技术知识点。

废话就不多说了,接下来好好看下这份资料。

如果你是一名Android、Java、前端等等开发人员,想要转入鸿蒙方向发展。可以直接领取这份资料辅助你的学习。鸿蒙OpenHarmony知识←前往。下面是鸿蒙开发的学习路线图。

针对鸿蒙成长路线打造的鸿蒙学习文档。鸿蒙(OpenHarmony )学习手册(共计1236页)与鸿蒙(OpenHarmony )开发入门教学视频,帮助大家在技术的道路上更进一步。

其中内容包含:

《鸿蒙开发基础》鸿蒙OpenHarmony知识←前往

  1. ArkTS语言
  2. 安装DevEco Studio
  3. 运用你的第一个ArkTS应用
  4. ArkUI声明式UI开发
  5. .……

《鸿蒙开发进阶》鸿蒙OpenHarmony知识←前往

  1. Stage模型入门
  2. 网络管理
  3. 数据管理
  4. 电话服务
  5. 分布式应用开发
  6. 通知与窗口管理
  7. 多媒体技术
  8. 安全技能
  9. 任务管理
  10. WebGL
  11. 国际化开发
  12. 应用测试
  13. DFX面向未来设计
  14. 鸿蒙系统移植和裁剪定制
  15. ……

《鸿蒙开发实战》鸿蒙OpenHarmony知识←前往

  1. ArkTS实践
  2. UIAbility应用
  3. 网络案例
  4. ……

最后

鸿蒙是完全具备无与伦比的机遇和潜力的;预计到年底将有 5,000 款的应用完成原生鸿蒙开发,这么多的应用需要开发,也就意味着需要有更多的鸿蒙人才。鸿蒙开发工程师也将会迎来爆发式的增长,学习鸿蒙势在必行!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/727.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SQL增加主键约束的条件

结论 常见认为设为主键的条件为&#xff1a; 值唯一不含空值 具体实施中会出现各种问题 添加主键约束的条件细则&#xff1a; 值唯一数据中不含空值在定义时需要not null约束&#xff08;使用check约束不行&#xff09; 验证实验 接下来我做了关于这个细则的验证实验&am…

【MATLAB源码-第193期】基于matlab的网络覆盖率NOA优化算法仿真对比VFINOA,VFPSO,VFNGO,VFWOA等算法。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 NOA&#xff08;Network Optimization Algorithm&#xff0c;网络优化算法&#xff09;是一个针对网络覆盖率优化的算法&#xff0c;它主要通过优化网络中节点的分布和配置来提高网络的整体覆盖性能。网络覆盖率是衡量一个无…

【学习】软件压力测试对软件产品的作用

在信息化高速发展的今天&#xff0c;软件产品已经成为各行各业不可或缺的一部分。然而&#xff0c;随着软件功能的日益复杂和用户需求的不断增长&#xff0c;软件产品的稳定性和可靠性问题也愈发凸显。在这样的背景下&#xff0c;软件压力测试作为软件质量保障的重要手段之一&a…

【项目亮点】大厂中分布式事务的最佳实践 问题产生->难点与权衡(偏爱Saga)->解决方案

【项目亮点】大厂中分布式事务的最佳实践 问题产生->难点与权衡->解决方案->底层实现->应用案例 不断有同学问我大厂中实践分布式事务的问题,这里从分布式事务的产生,到强弱一致性与性能的权衡,再到最终落地的解决方案,再到实际的代码实现,再到我工作中实际使用SA…

【C语言__动态内存管理__复习篇6】

目录 前言 一、动态内存管理 二、动态内存函数 2.1 malloc 2.2 free 2.3 calloc 2.4 realloc 三、动态内存常见的6个使用错误 3.1 接收malloc/calloc返回的参数后未及时检查是否为NULL 3.2 越界访问动态内存空间 3.3 对非动态开辟的内存使用free释放 3.4 使用free只释放了…

AI时代,我要如何学习,才能跟上步伐

在21世纪这个被数据驱动的时代&#xff0c;人工智能&#xff08;AI&#xff09;已经渗透到我们生活的方方面面。无论是智能手机中的语音助手、在线客服的聊天机器人&#xff0c;还是自动驾驶汽车&#xff0c;AI的应用都在告诉我们一个信息&#xff1a;未来已来。因此&#xff0…

1.微服务介绍

完整的微服务架构图 注册中心 配置中心 服务集群 服务网关 分布式缓存 分布式搜索 数据库集群 消息队列 分布式日志服务 系统监控链路追踪 Jenkins docker k8s 技术栈 微服务治理&#xff1a; 注册发现、远程调用、负载均衡、配置管理、网关路由、系统保护、流量…

企业单位IPTV数字电视直播与点播系统-中国卫通怀来地球站IPTV数字电视直播与点播系统应用浅析

企业单位IPTV数字电视直播与点播系统-中国卫通怀来地球站IPTV数字电视直播与点播系统应用浅析 由北京海特伟业科技有限公司任洪卓发布于2024年4月19日 一、运营商光猫接入企业/单位IPTV数字电视直播与点播系统建设概述 中国卫通怀来地球站&#xff0c;位于怀来县土木镇&#xf…

小球反弹(蓝桥杯)

文章目录 小球反弹【问题描述】答案&#xff1a;1100325199.77解题思路模拟 小球反弹 【问题描述】 有一长方形&#xff0c;长为 343720 单位长度&#xff0c;宽为 233333 单位长度。在其内部左上角顶点有一小球&#xff08;无视其体积&#xff09;&#xff0c;其初速度如图所…

CentOS 7静默安装Oracle 11g(记一次最小化CentOS 7安装Oracle 11g的经历)

# [pdf在线免费转word文档](https://orcc.online/pdf) https://orcc.online/pdf 1.最小化安装CentOS 7后首先设置一下固定IP 可以先查询一下自己的网卡设备的名称&#xff0c;是ens33&#xff0c;所以网卡配置文件名称就是ifcfg-ens33&#xff08;前面的ifcfg-不用管&#xf…

HCIP-Datacom-ARST必选题库_01_ACL【7道题】

一、单选 1.下面是一台路由器的部分配置,关于该配置描述正确的是&#xff1a; 源地址为1.1.1.1的数据包匹配第一条ACL语句rule 0,匹配规则为允许 源地址为1.1.1.3的数据包匹配第三条ACL语句rule 2,匹配规则为拒绝 源地址为1.1.1.4的数据包匹配第四条ACL语句rule 3,匹配规则为允…

【Python】函数(纯干货版)

目录 什么是函数 函数定义 函数的文档说明 局部变量和全局变量 综合案例&#xff1a;模拟实现ATM界面 什么是函数 函数是组织好的&#xff0c;可重复使用的&#xff0c;用于实现特定功能的代码段&#xff0c;将功能封装在函数内&#xff0c;可供随时随地重复利用&#xff…

加固平板电脑加速国产化!应用场景大科普

随着时代的发展&#xff0c;国产化平板电脑已经开始慢慢的实现了&#xff0c;国产芯片、国产操作系统、国产 GPU、国产 GNSS 等众多关键技术的集成与应用。全新的平台&#xff0c;全新的突破&#xff0c;全新的应用&#xff0c;让国产加固型移动计算机系列往前进了一步。那么加…

如何用 AI 工具做数据分析与可视化?

&#xff08;注&#xff1a;本文为小报童精选文章。已订阅小报童或加入知识星球「玉树芝兰」用户请勿重复付费&#xff09; 万字长文&#xff0c;助力你用 AI 提升科研效率。 2024 年 4 月 14 日&#xff0c;应武汉大学信息管理学院的邀请&#xff0c;我和北京大学步一老师给几…

python 头文件怎么写

本文主要以python2为例。首先介绍一下Python头文件的编程风格&#xff0c;然后再给大家详细介绍import部分的基本用法。这两个部分就是Python中头文件的组成模块。 编程风格 #!/usr/bin/env python #在文件头部 ( 第一行 ) 加上 设置 Python 解释器 # -*- coding: utf…

【开源】使用Python+Flask+Mysql快速开发一个用户增删改查系统

项目演示 项目本身很简单&#xff0c;增删改查是几乎所有系统的骨架。正所谓万丈高楼平地起&#xff0c;学会了增删改查&#xff0c;航母就指日可待了&#xff1a;&#xff09;&#xff0c;光速入门&#xff0c;直接看演示图&#xff1a; 项目地址 https://github.com/mudf…

设计模式代码实战-责任链模式

1、问题描述 小明所在的公司请假需要在OA系统上发布申请&#xff0c;整个请求流程包括多个处理者&#xff0c;每个处理者负责处理不同范围的请假天数&#xff0c;如果一个处理者不能处理请求&#xff0c;就会将请求传递给下一个处理者&#xff0c;请你实现责任链模式&#xff…

大华相机C#学习之IDevice类

获取方式 Enumerator.GetDeviceByGigeIP() 通过IP地址获取设备对象。 private void test_Click(object sender, EventArgs e) {devicesEnumerator.EnumerateDevices();device Enumerator.GetDeviceByGigeIP("192.168.0.11"); } 常用属性 DeviceInfo 获取设备的信…

uniapp_微信小程序_预约时间组件的使用

一、官方文档 DatetimePicker 选择器 | uView 2.0 - 全面兼容 nvue 的 uni-app 生态框架 - uni-app UI 框架 (uviewui.com) 二、完成的效果 之前使用的是Calendar 日历 这个太耗性能了&#xff0c;直接页面卡顿&#xff0c;所以就换成以上选择器了 三、代码 <u-datetime-p…

短信登录session-redis

1.流程 1.1 发送验证码 模拟路径 http://127.0.0.1:8080/api/user/code?phone1335566 Request Method:POSTcontroller层 /*** 发送手机验证码*/PostMapping("code")public Result sendCode(RequestParam("phone") String phone, HttpSession session) {…