当前位置:   article > 正文

【开源】嵌入式Linux(IMX6U)应用层综合项目(1)--云平台调试APP

【开源】嵌入式Linux(IMX6U)应用层综合项目(1)--云平台调试APP

目录

1.简介

1.1功能介绍

1.2技术栈介绍

1.3演示视频

1.4硬件介绍

2.软件设计

2.1连接阿里云

2.2云平台调试UI

2.3Ui_main.c界面切换处理文件

2.4.main函数

3.结尾(附网盘链接)


1.简介

此文章并不是教程,只能当作笔者的学习分享,只会做一些简单的介绍,其他的各位结合着代码和运行现象自己分析吧,相信通过函数名和注释,基本上是不难看懂代码的,其中涉及到的一些技术栈,也请各位学习到的时候多查阅资料。

本篇的内容为嵌入式Linux应用层的一个综合性比较强的项目,结尾会将源码放在网盘中开源出来,笔者能力有限,只是简单的把功能实现了,代码开源供大家一起交流学习,有什么好的建议,请各位一定不吝赐教!!!

1.1功能介绍

项目包括了四个app:

1.云平台的调试窗口,用于查看订阅主题所下发的数据,另一个为输入Json格式的数据来控制STM32单片机上的外设。

2.智能家居的界面,有4个图片按钮用于控制STM32板子上的LED灯、门(舵机)、蜂鸣器,量计分别为温度、湿度和亮度的值,同样是STM32获取发布到云平台的。

3.通过一个摄像头模块做的一个相机功能,可以拍照、录像,以及查看拍摄的照片,和播放录制视频的回放。

4.简易的音乐播放器:能够切换歌曲,以及暂停播放音乐。

1.2技术栈介绍

虽然项目简单,但是所涉及到的技术栈还是比较杂,我简单在此列出:

 1.LVGL库用于绘制UI。

2.MQTT协议,连接阿里云平台与STM32通讯。

3.alsa库用于音频处理。

4.LED、BEEP

5.V4L2 摄像头应用编程

1.3演示视频

【开源】嵌入式Linux应用层物联网小项目|通过MQTT协议与STM32通讯_哔哩哔哩_bilibili

1.4硬件介绍

硬件使用的是正点原子的阿尔法开发板,芯片是IMX6U,类似开发板应该都可以运行。

2.软件设计

2.1连接阿里云

要注意的部分就是要让开发板连上网,然后将三元组替换为自己所创建产品和设备对应的三元组。如何创建就不在本篇中赘述了,不清楚的可以看看之前发的文章:一篇文章将带你从0到1让Linux系统连接阿里云--MQTT协议【傻瓜式教程】_linux用mqtt连接云-CSDN博客

本文件干的事,就是让开发板连上云平台,然后通过订阅了STM32下发数据的主题,在回调函数中处理STM32下发的传感器数据。

  1. /*
  2. * 这个例程适用于`Linux`这类支持pthread的POSIX设备, 它演示了用SDK配置MQTT参数并建立连接, 之后创建2个线程
  3. *
  4. * + 一个线程用于保活长连接
  5. * + 一个线程用于接收消息, 并在有消息到达时进入默认的数据回调, 在连接状态变化时进入事件回调
  6. *
  7. * 需要用户关注或修改的部分, 已经用 TODO 在注释中标明
  8. *
  9. */
  10. #include <stdio.h>
  11. #include <string.h>
  12. #include <unistd.h>
  13. #include <pthread.h>
  14. #include "aiot_state_api.h"
  15. #include "aiot_sysdep_api.h"
  16. #include "aiot_mqtt_api.h"
  17. #include "mqtt_aliyun.h"
  18. #include "cJSON.h"
  19. #include "ui_main.h"
  20. /* TODO: 替换为自己设备的三元组 */
  21. char *product_key = "k1h2hJkoTA7";
  22. char *device_name = "Linux_ATK";
  23. char *device_secret = "00708c34c4e4b75f035c07e13613285f";
  24. char *mqtt_host = "iot-06z00jlxn39wea0.mqtt.iothub.aliyuncs.com";
  25. const uint16_t port = 8883;
  26. //const uint16_t port = 1883;
  27. /* 位于portfiles/aiot_port文件夹下的系统适配函数集合 */
  28. extern aiot_sysdep_portfile_t g_aiot_sysdep_portfile;
  29. /* 位于external/ali_ca_cert.c中的服务器证书 */
  30. extern const char *ali_ca_cert;
  31. static pthread_t g_mqtt_process_thread = NULL;
  32. static pthread_t g_mqtt_recv_thread = NULL;
  33. static uint8_t g_mqtt_process_thread_running = 0;
  34. static uint8_t g_mqtt_recv_thread_running = 0;
  35. extern g_sensor_t g_sensor;
  36. extern lv_obj_t * temp_bar;
  37. lv_anim_t a;
  38. //char *sub_topic1 = "/k1h2hJkoTA7/Linux_ATK/user/get";
  39. char *sub_topic = "/sys/k1h2hJkoTA7/Linux_ATK/thing/service/property/set";
  40. // 比较主题的函数
  41. int compare_topic(const char* received_topic, const char* sub_topic, size_t sub_topic_len) {
  42. // 确保接收到的主题长度至少与订阅的主题长度相同
  43. if (strlen(received_topic) < sub_topic_len) {
  44. return 0; // 主题长度不足,无法比较
  45. }
  46. // 截取接收到的主题的前缀部分
  47. char received_prefix[sub_topic_len + 1];
  48. strncpy(received_prefix, received_topic, sub_topic_len);
  49. received_prefix[sub_topic_len] = '\0'; // 确保截取的字符串以 null 结尾
  50. printf("received_prefix = %s\r\n", received_prefix);
  51. // 比较前缀
  52. return strcmp(received_prefix, sub_topic) == 0;
  53. }
  54. void Get_Data_From_Cloud(char *payload, int payload_len)
  55. {
  56. char buf[25];
  57. // 根据需要解析 JSON 数据
  58. cJSON *json = cJSON_Parse(payload);
  59. if (json != NULL)
  60. {
  61. // 解析并处理 params 数据
  62. cJSON *params = cJSON_GetObjectItem(json, "params");
  63. if (cJSON_IsObject(params))
  64. {
  65. // 解析并处理 Temp 数据
  66. cJSON *Temp = cJSON_GetObjectItem(params, "Temp");
  67. if (cJSON_IsNumber(Temp))
  68. {
  69. printf("Temp: %d\n", Temp->valueint);
  70. // 在这里处理 Temp 数据
  71. g_sensor.temp_value = Temp->valueint;
  72. lv_snprintf(buf, 25, "Temp:%d\n", g_sensor.temp_value);
  73. ui_update_sublish_message(buf, payload_len);
  74. memset(buf, 0, 25);
  75. }
  76. // 解析并处理 Humi 数据
  77. cJSON *humi = cJSON_GetObjectItem(params, "Humi");
  78. if (cJSON_IsNumber(humi))
  79. {
  80. printf("Humi: %d\n", humi->valueint);
  81. // 在这里处理 Humi 数据
  82. g_sensor.humi_value = humi->valueint;
  83. lv_snprintf(buf, 25, "Humi:%d\n", g_sensor.humi_value);
  84. ui_update_sublish_message(buf, payload_len);
  85. memset(buf, 0, 25);
  86. }
  87. // 解析并处理 light 数据
  88. cJSON *light = cJSON_GetObjectItem(params, "light");
  89. if (cJSON_IsNumber(light))
  90. {
  91. printf("light: %d\n", light->valueint);
  92. // 在这里处理 light 数据
  93. g_sensor.light_value = light->valueint;
  94. lv_snprintf(buf, 25, "Light:%d\n", g_sensor.light_value);
  95. ui_update_sublish_message(buf, payload_len);
  96. memset(buf, 0, 25);
  97. }
  98. }
  99. else
  100. {
  101. printf("Error: params is not an object\n");
  102. }
  103. cJSON_Delete(json);
  104. }
  105. else
  106. {
  107. printf("Error: Unable to parse JSON\n");
  108. }
  109. }
  110. int mqtt_payload_handle(char *payload, int payload_len)
  111. {
  112. if(payload_len > 0)
  113. {
  114. payload[payload_len] = '\0';
  115. }
  116. ui_update_sublish_message(payload, payload_len);
  117. if(!strcmp(payload, "led_on"))
  118. {
  119. set_led_mode("on");
  120. return LED_ON;
  121. }
  122. else if(!strcmp(payload, "led_off"))
  123. {
  124. set_led_mode("off");
  125. return LED_OFF;
  126. }
  127. else if(!strcmp(payload, "beep_on"))
  128. {
  129. set_beep_mode("on");
  130. return BEEP_ON;
  131. }
  132. else if(!strcmp(payload, "beep_off"))
  133. {
  134. set_beep_mode("off");
  135. return BEEP_OFF;
  136. }
  137. return 0;
  138. }
  139. /* TODO: 如果要关闭日志, 就把这个函数实现为空, 如果要减少日志, 可根据code选择不打印
  140. *
  141. * 例如: [1577589489.033][LK-0317] mqtt_basic_demo&gb80sFmX7yX
  142. *
  143. * 上面这条日志的code就是0317(十六进制), code值的定义见core/aiot_state_api.h
  144. *
  145. */
  146. /* 日志回调函数, SDK的日志会从这里输出 */
  147. int32_t demo_state_logcb(int32_t code, char *message)
  148. {
  149. //printf("%s", message);
  150. //return 0;
  151. }
  152. /* MQTT事件回调函数, 当网络连接/重连/断开时被触发, 事件定义见core/aiot_mqtt_api.h */
  153. void demo_mqtt_event_handler(void *handle, const aiot_mqtt_event_t *event, void *userdata)
  154. {
  155. switch (event->type) {
  156. /* SDK因为用户调用了aiot_mqtt_connect()接口, 与mqtt服务器建立连接已成功 */
  157. case AIOT_MQTTEVT_CONNECT: {
  158. printf("AIOT_MQTTEVT_CONNECT\n");
  159. /* TODO: 处理SDK建连成功, 不可以在这里调用耗时较长的阻塞函数 */
  160. }
  161. break;
  162. /* SDK因为网络状况被动断连后, 自动发起重连已成功 */
  163. case AIOT_MQTTEVT_RECONNECT: {
  164. printf("AIOT_MQTTEVT_RECONNECT\n");
  165. /* TODO: 处理SDK重连成功, 不可以在这里调用耗时较长的阻塞函数 */
  166. }
  167. break;
  168. /* SDK因为网络的状况而被动断开了连接, network是底层读写失败, heartbeat是没有按预期得到服务端心跳应答 */
  169. case AIOT_MQTTEVT_DISCONNECT: {
  170. char *cause = (event->data.disconnect == AIOT_MQTTDISCONNEVT_NETWORK_DISCONNECT) ? ("network disconnect") :
  171. ("heartbeat disconnect");
  172. printf("AIOT_MQTTEVT_DISCONNECT: %s\n", cause);
  173. /* TODO: 处理SDK被动断连, 不可以在这里调用耗时较长的阻塞函数 */
  174. }
  175. break;
  176. default: {
  177. }
  178. }
  179. }
  180. /* MQTT默认消息处理回调, 当SDK从服务器收到MQTT消息时, 且无对应用户回调处理时被调用 */
  181. void demo_mqtt_default_recv_handler(void *handle, const aiot_mqtt_recv_t *packet, void *userdata)
  182. {
  183. switch (packet->type) {
  184. case AIOT_MQTTRECV_HEARTBEAT_RESPONSE: {
  185. printf("heartbeat response\n");
  186. /* TODO: 处理服务器对心跳的回应, 一般不处理 */
  187. }
  188. break;
  189. case AIOT_MQTTRECV_SUB_ACK: {
  190. printf("suback, res: -0x%04X, packet id: %d, max qos: %d\n",
  191. -packet->data.sub_ack.res, packet->data.sub_ack.packet_id, packet->data.sub_ack.max_qos);
  192. /* TODO: 处理服务器对订阅请求的回应, 一般不处理 */
  193. }
  194. break;
  195. /* TODO: 云-->客户端 处理服务器下发的业务报文 */
  196. case AIOT_MQTTRECV_PUB: {
  197. //printf("pub, qos: %d, topic: %.*s\n", packet->data.pub.qos, packet->data.pub.topic_len, packet->data.pub.topic);
  198. //printf("pub, payload: %.*s\n", packet->data.pub.payload_len, packet->data.pub.payload);
  199. /* TODO: 处理服务器下发的业务报文 */
  200. printf("payload = %.*s\n", packet->data.pub.payload_len, packet->data.pub.payload);
  201. //printf("packet->data.pub.topic = %s\n", packet->data.pub.topic);
  202. Get_Data_From_Cloud(packet->data.pub.payload, packet->data.pub.payload_len);
  203. }
  204. break;
  205. case AIOT_MQTTRECV_PUB_ACK: {
  206. printf("puback, packet id: %d\n", packet->data.pub_ack.packet_id);
  207. /* TODO: 处理服务器对QoS1上报消息的回应, 一般不处理 */
  208. }
  209. break;
  210. default: {
  211. }
  212. }
  213. }
  214. /* 执行aiot_mqtt_process的线程, 包含心跳发送和QoS1消息重发 */
  215. void *demo_mqtt_process_thread(void *args)
  216. {
  217. int32_t res = STATE_SUCCESS;
  218. while (g_mqtt_process_thread_running) {
  219. res = aiot_mqtt_process(args);
  220. if (res == STATE_USER_INPUT_EXEC_DISABLED) {
  221. break;
  222. }
  223. sleep(1);
  224. }
  225. return 0;
  226. }
  227. /* 执行aiot_mqtt_recv的线程, 包含网络自动重连和从服务器收取MQTT消息 */
  228. void *demo_mqtt_recv_thread(void *args)
  229. {
  230. int32_t res = STATE_SUCCESS;
  231. while (g_mqtt_recv_thread_running) {
  232. res = aiot_mqtt_recv(args);
  233. if (res < STATE_SUCCESS) {
  234. if (res == STATE_USER_INPUT_EXEC_DISABLED) {
  235. break;
  236. }
  237. sleep(1);
  238. }
  239. }
  240. return 0;
  241. }
  242. void *mqtt_handle = NULL;
  243. int net_mqtt_aliyun_init(void)
  244. {
  245. int32_t res = STATE_SUCCESS;
  246. aiot_sysdep_network_cred_t cred; /* 安全凭据结构体, 如果要用TLS, 这个结构体中配置CA证书等参数 */
  247. /* 配置SDK的底层依赖 */
  248. aiot_sysdep_set_portfile(&g_aiot_sysdep_portfile);
  249. /* 配置SDK的日志输出 */
  250. aiot_state_set_logcb(demo_state_logcb);
  251. /* 创建SDK的安全凭据, 用于建立TLS连接 */
  252. memset(&cred, 0, sizeof(aiot_sysdep_network_cred_t));
  253. cred.option = AIOT_SYSDEP_NETWORK_CRED_SVRCERT_CA; /* 使用RSA证书校验MQTT服务端 */
  254. cred.max_tls_fragment = 16384; /* 最大的分片长度为16K, 其它可选值还有4K, 2K, 1K, 0.5K */
  255. cred.sni_enabled = 1; /* TLS建连时, 支持Server Name Indicator */
  256. cred.x509_server_cert = ali_ca_cert; /* 用来验证MQTT服务端的RSA根证书 */
  257. cred.x509_server_cert_len = strlen(ali_ca_cert); /* 用来验证MQTT服务端的RSA根证书长度 */
  258. /* 创建1个MQTT客户端实例并内部初始化默认参数 */
  259. mqtt_handle = aiot_mqtt_init();
  260. if (mqtt_handle == NULL) {
  261. printf("aiot_mqtt_init failed\n");
  262. return -1;
  263. }
  264. /* TODO: 如果以下代码不被注释, 则例程会用TCP而不是TLS连接云平台 */
  265. /*
  266. {
  267. memset(&cred, 0, sizeof(aiot_sysdep_network_cred_t));
  268. cred.option = AIOT_SYSDEP_NETWORK_CRED_NONE;
  269. }
  270. */
  271. /* 配置MQTT服务器地址 */
  272. aiot_mqtt_setopt(mqtt_handle, AIOT_MQTTOPT_HOST, (void *)mqtt_host);
  273. /* 配置MQTT服务器端口 */
  274. aiot_mqtt_setopt(mqtt_handle, AIOT_MQTTOPT_PORT, (void *)&port);
  275. /* 配置设备productKey */
  276. aiot_mqtt_setopt(mqtt_handle, AIOT_MQTTOPT_PRODUCT_KEY, (void *)product_key);
  277. /* 配置设备deviceName */
  278. aiot_mqtt_setopt(mqtt_handle, AIOT_MQTTOPT_DEVICE_NAME, (void *)device_name);
  279. /* 配置设备deviceSecret */
  280. aiot_mqtt_setopt(mqtt_handle, AIOT_MQTTOPT_DEVICE_SECRET, (void *)device_secret);
  281. /* 配置网络连接的安全凭据, 上面已经创建好了 */
  282. aiot_mqtt_setopt(mqtt_handle, AIOT_MQTTOPT_NETWORK_CRED, (void *)&cred);
  283. /* 配置MQTT默认消息接收回调函数 */
  284. aiot_mqtt_setopt(mqtt_handle, AIOT_MQTTOPT_RECV_HANDLER, (void *)demo_mqtt_default_recv_handler);
  285. /* 配置MQTT事件回调函数 */
  286. aiot_mqtt_setopt(mqtt_handle, AIOT_MQTTOPT_EVENT_HANDLER, (void *)demo_mqtt_event_handler);
  287. /* 与服务器建立MQTT连接 */
  288. res = aiot_mqtt_connect(mqtt_handle);
  289. if (res < STATE_SUCCESS) {
  290. /* 尝试建立连接失败, 销毁MQTT实例, 回收资源 */
  291. aiot_mqtt_deinit(&mqtt_handle);
  292. printf("aiot_mqtt_connect failed: -0x%04X\n\r\n", -res);
  293. printf("please check variables like mqtt_host, produt_key, device_name, device_secret in demo\r\n");
  294. return -1;
  295. }
  296. /* MQTT 订阅topic功能示例, 请根据自己的业务需求进行使用 */
  297. {
  298. res = aiot_mqtt_sub(mqtt_handle, sub_topic, NULL, 1, NULL);
  299. if (res < 0) {
  300. printf("aiot_mqtt_sub failed, res: -0x%04X\n", -res);
  301. return -1;
  302. }
  303. }
  304. /* MQTT 发布消息功能示例, 请根据自己的业务需求进行使用 */
  305. // {
  306. // char *pub_topic = "/k1h2hJkoTA7/Linux_ATK/user/update";
  307. // char *pub_payload = "{\"id\":\"1\",\"version\":\"1.0\",\"params\":{\"LightSwitch\":0}}";
  308. // res = aiot_mqtt_pub(mqtt_handle, pub_topic, (uint8_t *)pub_payload, (uint32_t)strlen(pub_payload), 0);
  309. // if (res < 0) {
  310. // printf("aiot_mqtt_sub failed, res: -0x%04X\n", -res);
  311. // return -1;
  312. // }
  313. // }
  314. /* 创建一个单独的线程, 专用于执行aiot_mqtt_process, 它会自动发送心跳保活, 以及重发QoS1的未应答报文 */
  315. g_mqtt_process_thread_running = 1;
  316. res = pthread_create(&g_mqtt_process_thread, NULL, demo_mqtt_process_thread, mqtt_handle);
  317. if (res < 0) {
  318. printf("pthread_create demo_mqtt_process_thread failed: %d\n", res);
  319. return -1;
  320. }
  321. printf("pthread_create demo_mqtt_process_thread successfully!\n");
  322. /* 创建一个单独的线程用于执行aiot_mqtt_recv, 它会循环收取服务器下发的MQTT消息, 并在断线时自动重连 */
  323. g_mqtt_recv_thread_running = 1;
  324. res = pthread_create(&g_mqtt_recv_thread, NULL, demo_mqtt_recv_thread, mqtt_handle);
  325. if (res < 0) {
  326. printf("pthread_create demo_mqtt_recv_thread failed: %d\n", res);
  327. //lv_textarea_set_text(recv_ta, "aliyun unconnected!");
  328. return -1;
  329. }
  330. //lv_textarea_set_text(recv_ta, "aliyun connected!");
  331. // /* 断开MQTT连接, 一般不会运行到这里 */
  332. // g_mqtt_process_thread_running = 0;
  333. // g_mqtt_recv_thread_running = 0;
  334. // sleep(1);
  335. // pthread_join(g_mqtt_process_thread, NULL);
  336. // pthread_join(g_mqtt_recv_thread, NULL);
  337. // res = aiot_mqtt_disconnect(mqtt_handle);
  338. // if (res < STATE_SUCCESS) {
  339. // aiot_mqtt_deinit(&mqtt_handle);
  340. // printf("aiot_mqtt_disconnect failed: -0x%04X\n", -res);
  341. // return -1;
  342. // }
  343. // /* 销毁MQTT实例, 一般不会运行到这里 */
  344. // res = aiot_mqtt_deinit(&mqtt_handle);
  345. // if (res < STATE_SUCCESS) {
  346. // printf("aiot_mqtt_deinit failed: -0x%04X\n", -res);
  347. // return -1;
  348. // }
  349. return 0;
  350. }
  351. int net_mqtt_aliyun_thread_delete(void)
  352. {
  353. if(pthread_join(g_mqtt_process_thread, NULL) != 0)
  354. {
  355. printf("delete g_mqtt_process_thread false\n");
  356. return -1;
  357. }
  358. if(pthread_join(g_mqtt_recv_thread, NULL) != 0)
  359. {
  360. printf("delete g_mqtt_recv_thread false\n");
  361. return -1;
  362. }
  363. return 0;
  364. }

2.2云平台调试UI

大致内容分为两部分,用lvgl创建了两个框,一个只接收显示STM32采集到的传感器数据,另一个框点击会出现一个键盘,可以输入对应的Json格式数据去控制STM32的外设。这一块就是利用了MQTT的订阅和发布主题的功能,不了解的可以自行百度。

  1. #include "ui_app_yunpintai.h"
  2. lv_obj_t * recv_ta;
  3. static lv_obj_t * kb;
  4. extern void *mqtt_handle;
  5. extern char *pub_topic;
  6. /*打印云下发的payload 到Sublish栏中*/
  7. void ui_update_sublish_message(char *payload, int payload_len)
  8. {
  9. if(payload_len > 25){
  10. payload_len = 25;
  11. }
  12. char buf[26];
  13. strncpy(buf, payload, payload_len);
  14. buf[payload_len] = '\0';
  15. printf("buf = %s\n", buf);
  16. if(recv_ta == NULL)
  17. {
  18. printf("Error: recv_ta is NULL\n");
  19. return;
  20. }
  21. lv_textarea_add_text(recv_ta, buf);
  22. }
  23. static void publish_event_cb(lv_event_t * e)
  24. {
  25. int res;
  26. lv_event_code_t code = lv_event_get_code(e);
  27. lv_obj_t * ta = lv_event_get_target(e);
  28. if (code == LV_EVENT_FOCUSED)
  29. {
  30. lv_keyboard_set_textarea(kb, ta); // Set the keyboard to the focused text area
  31. lv_obj_clear_flag(kb, LV_OBJ_FLAG_HIDDEN); // Show the keyboard
  32. }
  33. else if (code == LV_EVENT_DEFOCUSED)
  34. {
  35. lv_obj_add_flag(kb, LV_OBJ_FLAG_HIDDEN); // Hide the keyboard
  36. }
  37. else if(code == LV_EVENT_READY)
  38. {
  39. //char *pub_payload = "{\"LED\":1}";
  40. const char * text = lv_textarea_get_text(ta);
  41. printf("Sent text: %s\n", text);
  42. res = aiot_mqtt_pub(mqtt_handle, pub_topic, (uint8_t *)text, (uint32_t)strlen(text), 0);
  43. if (res < 0) {
  44. printf("aiot_mqtt_sub failed, res: -0x%04X\n", -res);
  45. return -1;
  46. }
  47. }
  48. }
  49. void ui_app_yunpintai(void)
  50. {
  51. //ui_app_clear_area();
  52. clear_area(0, 0, 800, 480, lv_color_hex(MY_UI_COLOR_DEEP_WHITE));
  53. ui_left_app_bar(20, 70);
  54. lv_obj_t * img = lv_img_create(lv_scr_act());
  55. lv_img_set_src(img, &img_yunpintai_on);
  56. lv_obj_align(img, LV_ALIGN_TOP_LEFT, 32, 90);
  57. lv_obj_t * mid_label = lv_label_create(lv_scr_act());
  58. lv_label_set_text(mid_label, "Cloud");
  59. static lv_style_t mid_label_style;
  60. lv_style_init(&mid_label_style);
  61. lv_style_set_text_font(&mid_label_style, &lv_font_montserrat_24); // 设置字体
  62. lv_style_set_text_color(&mid_label_style, lv_color_hex(MY_UI_COLOR_BLACK)); // 设置字体颜色
  63. lv_obj_add_style(mid_label, &mid_label_style, LV_PART_MAIN);
  64. lv_obj_align(mid_label, LV_ALIGN_TOP_MID, 0, 20);
  65. /*创建Sub订阅信息栏*/
  66. lv_obj_t * sub_label = lv_label_create(lv_scr_act());
  67. lv_label_set_text(sub_label, "Sublish");
  68. static lv_style_t sub_label_style;
  69. lv_style_init(&sub_label_style);
  70. lv_style_set_text_font(&sub_label_style, &lv_font_montserrat_48); // 设置字体
  71. lv_style_set_text_color(&sub_label_style, lv_color_hex(MY_UI_COLOR_BLUE)); // 设置字体颜色
  72. lv_obj_add_style(sub_label, &sub_label_style, LV_PART_MAIN);
  73. lv_obj_align(sub_label, LV_ALIGN_TOP_LEFT, 195, 65);
  74. /*设置sub发布信息调试框*/
  75. /* Create the receive text area */
  76. recv_ta = lv_textarea_create(lv_scr_act());
  77. printf("recv_ta after create: %p\n", recv_ta);
  78. lv_textarea_set_one_line(recv_ta, false);
  79. lv_textarea_set_password_mode(recv_ta, false);
  80. lv_textarea_set_text(recv_ta, "");
  81. lv_obj_set_size(recv_ta, 320, 320);
  82. lv_obj_align(recv_ta, LV_ALIGN_TOP_LEFT, 128, 130);
  83. lv_obj_set_style_text_font(recv_ta, &lv_font_montserrat_24, 0);
  84. /*创建Pub发布信息栏*/
  85. lv_obj_t * pub_label = lv_label_create(lv_scr_act());
  86. lv_label_set_text(pub_label, "Publish");
  87. static lv_style_t pub_label_style;
  88. lv_style_init(&pub_label_style);
  89. lv_style_set_text_font(&pub_label_style, &lv_font_montserrat_48); // 设置字体
  90. lv_style_set_text_color(&pub_label_style, lv_color_hex(MY_UI_COLOR_BLUE)); // 设置字体颜色
  91. lv_obj_add_style(pub_label, &pub_label_style, LV_PART_MAIN);
  92. lv_obj_align(pub_label, LV_ALIGN_TOP_LEFT, 525, 65);
  93. /*设置Pub发布信息调试框*/
  94. /*Create the one-line mode text area*/
  95. lv_obj_t * pub_ta = lv_textarea_create(lv_scr_act());
  96. lv_textarea_set_one_line(pub_ta, false);
  97. lv_textarea_set_password_mode(pub_ta, false);
  98. lv_obj_add_event_cb(pub_ta, publish_event_cb, LV_EVENT_ALL, NULL);
  99. lv_obj_set_size(pub_ta, 320, 320);
  100. lv_obj_align(pub_ta, LV_ALIGN_TOP_LEFT, 462, 130);
  101. /*Create temp_anim keyboard*/
  102. kb = lv_keyboard_create(lv_scr_act());
  103. lv_obj_set_size(kb, LV_HOR_RES, LV_VER_RES / 2);
  104. lv_keyboard_set_textarea(kb, pub_ta);
  105. lv_obj_set_style_text_font(kb, &lv_font_dejavu_16_persian_hebrew, 0);
  106. lv_obj_set_style_text_font(pub_ta, &lv_font_montserrat_24, 0);
  107. lv_obj_add_flag(kb, LV_OBJ_FLAG_HIDDEN);
  108. }

2.3Ui_main.c界面切换处理文件

此文件中,创建了一个线程来处理我们项目中所有UI的处理,左边的app功能栏中的4个图片按钮用于切换4个不同功能的app,其他的诸如清屏,按键回调函数,各位自行看看即可。

  1. #include "ui_main.h"
  2. #include "ui_app_yunpintai.h"
  3. #include "ui_app_smarthome.h"
  4. #include "ui_app_camera.h"
  5. #include "ui_app_music.h"
  6. static app_imgbtn_t app_imgbtn[NUM_APPS];
  7. static int current_active_app = -1;
  8. static pthread_t g_ds_ui_page_thread = NULL;
  9. extern void *mqtt_handle;
  10. extern lv_obj_t *recv_ta;
  11. char *pub_topic = "/k1h2hJkoTA7/Linux_ATK/user/Android_STM32";
  12. int app_index = 0;
  13. // 创建一个与背景颜色相同的对象来覆盖指定区域
  14. void clear_area(lv_coord_t x, lv_coord_t y, lv_coord_t width, lv_coord_t height, lv_color_t value)
  15. {
  16. lv_obj_t *clear_obj = lv_obj_create(lv_scr_act()); // 创建一个对象
  17. lv_obj_remove_style_all(clear_obj); // 移除默认样式
  18. lv_obj_set_size(clear_obj, width, height); // 设置对象大小
  19. lv_obj_set_pos(clear_obj, x, y); // 设置对象位置
  20. lv_obj_set_style_bg_color(clear_obj, value, 0); // 设置对象背景颜色,假设背景是白色
  21. lv_obj_set_style_bg_opa(clear_obj, LV_OPA_COVER, 0); // 设置对象不透明度
  22. }
  23. void ui_draw_backgroud_bar(lv_coord_t x, lv_coord_t y, lv_coord_t width, lv_coord_t height, lv_color_t value)
  24. {
  25. static lv_style_t style;
  26. lv_style_init(&style);
  27. /*Set temp_anim background color and temp_anim radius*/
  28. lv_style_set_radius(&style, 25);
  29. lv_style_set_bg_opa(&style, LV_OPA_COVER);
  30. lv_style_set_bg_color(&style, value);
  31. /*Create an object with the new style*/
  32. lv_obj_t *obj = lv_obj_create(lv_scr_act());
  33. lv_obj_add_style(obj, &style, 0);
  34. lv_obj_set_size(obj, width, height);
  35. lv_obj_align(obj, LV_ALIGN_TOP_LEFT, x, y);
  36. }
  37. /* 功能栏按键 切换app */
  38. static void imgbtn_event_handler(lv_event_t *e)
  39. {
  40. lv_obj_t *imgbtn = lv_event_get_target(e);
  41. lv_event_code_t code = lv_event_get_code(e);
  42. if (code == LV_EVENT_CLICKED)
  43. {
  44. printf("Button clicked!\n");
  45. /* Find the app_index of the pressed button */
  46. for (int i = 0; i < NUM_APPS; i++)
  47. {
  48. if (app_imgbtn[i].imgbtn == imgbtn)
  49. {
  50. app_index = i;
  51. break;
  52. }
  53. }
  54. printf("app_index = %d\n", app_index);
  55. /* Update the active app */
  56. current_active_app = app_index;
  57. /* Call ui_smarthome_page to update the UI */
  58. ui_page_main();
  59. }
  60. }
  61. void ui_create_app_imagebtn(unsigned int x, unsigned int y)
  62. {
  63. /* Define the image resources for each app button */
  64. app_imgbtn[0].img_on = &img_yunpintai_on;
  65. app_imgbtn[0].img_off = &img_yunpintai_off;
  66. app_imgbtn[1].img_on = &img_smarthome_on;
  67. app_imgbtn[1].img_off = &img_smarthome_off;
  68. app_imgbtn[2].img_on = &img_camera_on;
  69. app_imgbtn[2].img_off = &img_camera_off;
  70. app_imgbtn[3].img_on = &img_music_on;
  71. app_imgbtn[3].img_off = &img_music_off;
  72. /* Create an image button */
  73. for (int i = 0; i < NUM_APPS; i++)
  74. {
  75. app_imgbtn[i].imgbtn = lv_imgbtn_create(lv_scr_act());
  76. lv_imgbtn_set_src(app_imgbtn[i].imgbtn, LV_IMGBTN_STATE_RELEASED, NULL, NULL, app_imgbtn[i].img_off);
  77. lv_imgbtn_set_src(app_imgbtn[i].imgbtn, LV_IMGBTN_STATE_PRESSED, NULL, NULL, app_imgbtn[i].img_on);
  78. /* Add event handler to switch images on press */
  79. lv_obj_add_event_cb(app_imgbtn[i].imgbtn, imgbtn_event_handler, LV_EVENT_CLICKED, NULL);
  80. /* Set size and position */
  81. lv_obj_set_size(app_imgbtn[i].imgbtn, 64, 64);
  82. lv_obj_align(app_imgbtn[i].imgbtn, LV_ALIGN_TOP_LEFT, x + 12, y + 20 + 90 * i); // Adjust the positions as needed
  83. }
  84. }
  85. /*创建左app栏*/
  86. void ui_left_app_bar(unsigned int x, unsigned int y)
  87. {
  88. /* 创建左功能栏 */
  89. static lv_style_t style_bg;
  90. lv_style_init(&style_bg);
  91. lv_style_set_radius(&style_bg, 25);
  92. lv_style_set_bg_color(&style_bg, lv_color_hex(MY_UI_COLOR_WHITE));
  93. lv_style_set_bg_opa(&style_bg, LV_OPA_COVER);
  94. /* Create temp_anim background object */
  95. lv_obj_t *bg = lv_obj_create(lv_scr_act());
  96. lv_obj_add_style(bg, &style_bg, 0);
  97. lv_obj_set_size(bg, 90, 380);
  98. lv_obj_align(bg, LV_ALIGN_TOP_LEFT, x, y);
  99. ui_create_app_imagebtn(x, y);
  100. }
  101. static void destroy_previous_objects(void)
  102. {
  103. if (recv_ta != NULL)
  104. {
  105. lv_obj_del(recv_ta);
  106. recv_ta = NULL;
  107. printf("recv_ta destroyed\n");
  108. }
  109. }
  110. /* 页面主函数 */
  111. void ui_page_main(void)
  112. {
  113. destroy_previous_objects();
  114. lv_obj_set_style_bg_color(lv_scr_act(), lv_color_hex(MY_UI_COLOR_DEEP_WHITE), LV_PART_MAIN);
  115. ui_left_app_bar(20, 70);
  116. if (app_index == UI_APP_YUNPINTAI)
  117. {
  118. ui_app_yunpintai();
  119. }
  120. else if (app_index == UI_APP_SMARTHOME)
  121. {
  122. ui_app_smarthome();
  123. }
  124. else if (app_index == UI_APP_CAMERA)
  125. {
  126. ui_app_camera();
  127. }
  128. else if (app_index == UI_APP_MUSIC)
  129. {
  130. ui_app_music();
  131. }
  132. }
  133. void *ds_ui_page_thread(void *args)
  134. {
  135. ui_page_main();
  136. // setup_time_update();
  137. while (1)
  138. {
  139. lv_task_handler();
  140. usleep(5000);
  141. }
  142. return NULL;
  143. }
  144. int ui_app_page_init(void)
  145. {
  146. int res;
  147. res = pthread_create(&g_ds_ui_page_thread, NULL, ds_ui_page_thread, NULL);
  148. if (res != 0)
  149. {
  150. printf("pthread_create ds_ui_page_thread failed: %d\n", res);
  151. return -1;
  152. }
  153. printf("ds_ui_page_thread created successfully\n");
  154. return 0;
  155. }

2.4.main函数

在main函数中先需要初始化lvgl的相关内容,然后调用ui_app_page_init();和net_mqtt_aliyun_init();来对界面初始化和连接上阿里云。

  1. #include "app_main.h"
  2. extern lv_obj_t *recv_ta;
  3. #define DISP_BUF_SIZE (800 * 480)
  4. static void my_lvgl_init(void);
  5. int main(void)
  6. {
  7. printf("Hello Linux!!!\n\r");
  8. set_led_mode("off");
  9. my_lvgl_init();
  10. ui_app_page_init();
  11. net_mqtt_aliyun_init();
  12. while (1)
  13. {
  14. sleep(1);
  15. }
  16. return 0;
  17. }
  18. /*Set in lv_conf.h as `LV_TICK_CUSTOM_SYS_TIME_EXPR`*/
  19. uint32_t custom_tick_get(void)
  20. {
  21. static uint64_t start_ms = 0;
  22. if (start_ms == 0)
  23. {
  24. struct timeval tv_start;
  25. gettimeofday(&tv_start, NULL);
  26. start_ms = (tv_start.tv_sec * 1000000 + tv_start.tv_usec) / 1000;
  27. }
  28. struct timeval tv_now;
  29. gettimeofday(&tv_now, NULL);
  30. uint64_t now_ms;
  31. now_ms = (tv_now.tv_sec * 1000000 + tv_now.tv_usec) / 1000;
  32. uint32_t time_ms = now_ms - start_ms;
  33. return time_ms;
  34. }
  35. static void my_lvgl_init(void)
  36. {
  37. /*LittlevGL init*/
  38. lv_init();
  39. /*Linux frame buffer device init*/
  40. fbdev_init();
  41. /*A small buffer for LittlevGL to draw the screen's content*/
  42. static lv_color_t buf[DISP_BUF_SIZE];
  43. /*Initialize a descriptor for the buffer*/
  44. static lv_disp_draw_buf_t disp_buf;
  45. lv_disp_draw_buf_init(&disp_buf, buf, NULL, DISP_BUF_SIZE);
  46. /*Initialize and register a display driver*/
  47. static lv_disp_drv_t disp_drv;
  48. lv_disp_drv_init(&disp_drv);
  49. disp_drv.draw_buf = &disp_buf;
  50. disp_drv.flush_cb = fbdev_flush;
  51. disp_drv.hor_res = 800;
  52. disp_drv.ver_res = 480;
  53. lv_disp_drv_register(&disp_drv);
  54. /* Linux input device init */
  55. evdev_init();
  56. /* Initialize and register a display input driver */
  57. /* Segmentation fault:*/
  58. lv_indev_drv_t indev_drv;
  59. lv_indev_drv_init(&indev_drv); /*Basic initialization*/
  60. indev_drv.type = LV_INDEV_TYPE_POINTER;
  61. indev_drv.read_cb = evdev_read;
  62. lv_indev_t *my_indev = lv_indev_drv_register(&indev_drv);
  63. /* Segmentation fault:*/
  64. // printf("LVGL_Init OK!\r\n");
  65. }

3.结尾(附网盘链接)

链接:百度网盘 请输入提取码

提取码:2jia

--来自百度网盘超级会员V5的分享

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/知新_RL/article/detail/923047
推荐阅读
相关标签
  

闽ICP备14008679号