msgbox_cache_lvgl.c 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412
  1. /*
  2. * Copyright (c) 2020 Actions Technology Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #define LOG_MODULE_CUSTOMER
  7. #include <os_common_api.h>
  8. #include <mem_manager.h>
  9. #include <msgbox_cache.h>
  10. #include <lvgl/lvgl.h>
  11. #include <assert.h>
  12. #include <sys/atomic.h>
  13. #include <sys/dlist.h>
  14. LOG_MODULE_REGISTER(msgbox, LOG_LEVEL_INF);
  15. typedef struct {
  16. sys_dlist_t node;
  17. lv_obj_t * hwnd;
  18. void *user_data;
  19. const msgbox_dsc_t *dsc;
  20. uint16_t view_id;
  21. } msgbox_popup_t;
  22. typedef struct {
  23. const msgbox_dsc_t *dsc;
  24. uint8_t num;
  25. uint8_t en : 1;
  26. uint16_t top; /* top most msgbox popup id */
  27. atomic_t num_popups;
  28. sys_dlist_t popup_list;
  29. } msgbox_cache_ctx_t;
  30. static void _msgbox_popup_handler(uint16_t msgbox_id, uint8_t msg_id, void * msg_data, void * user_data);
  31. static const msgbox_dsc_t *_msgbox_find_dsc(uint16_t msgbox_id);
  32. static msgbox_popup_t *_msgbox_find_popup_by_id(uint16_t msgbox_id);
  33. static msgbox_popup_t *_msgbox_find_popup_by_hwnd(void * hwnd);
  34. static void _msgbox_update_top(void * hwnd);
  35. static void _call_view_proc(uint16_t view_id, uint16_t msgbox_id, uint8_t msg_id);
  36. static msgbox_cache_ctx_t msgbox_ctx;
  37. int msgbox_cache_init(const msgbox_dsc_t *dsc, uint8_t num)
  38. {
  39. memset(&msgbox_ctx, 0, sizeof(msgbox_ctx));
  40. msgbox_ctx.en = 1;
  41. msgbox_ctx.dsc = dsc;
  42. msgbox_ctx.num = num;
  43. msgbox_ctx.top = MSGBOX_ID_ALL;
  44. atomic_set(&msgbox_ctx.num_popups, 0);
  45. sys_dlist_init(&msgbox_ctx.popup_list);
  46. ui_manager_set_msgbox_popup_callback(_msgbox_popup_handler);
  47. return 0;
  48. }
  49. void msgbox_cache_deinit(void)
  50. {
  51. msgbox_cache_close(MSGBOX_ID_ALL, false);
  52. ui_manager_set_msgbox_popup_callback(NULL);
  53. }
  54. void msgbox_cache_set_en(bool en)
  55. {
  56. msgbox_ctx.en = en ? 1 : 0;
  57. }
  58. uint8_t msgbox_cache_num_popup_get(void)
  59. {
  60. return (uint8_t)atomic_get(&msgbox_ctx.num_popups);
  61. }
  62. uint16_t msgbox_cache_get_top(void)
  63. {
  64. return msgbox_ctx.top;
  65. }
  66. int msgbox_cache_popup(uint16_t msgbox_id, void *user_data)
  67. {
  68. if (msgbox_ctx.dsc == NULL || msgbox_ctx.en == 0) {
  69. SYS_LOG_WRN("msgbox %u disabled", msgbox_id);
  70. return -EPERM;
  71. }
  72. const msgbox_dsc_t *dsc = _msgbox_find_dsc(msgbox_id);
  73. if (!dsc || (dsc->handler == NULL && dsc->popup_cb == NULL)) {
  74. SYS_LOG_WRN("msgbox %u invalid", msgbox_id);
  75. return -EINVAL;
  76. }
  77. atomic_inc(&msgbox_ctx.num_popups);
  78. if (ui_msgbox_popup(msgbox_id, user_data)) {
  79. atomic_dec(&msgbox_ctx.num_popups);
  80. return -ENOMSG;
  81. }
  82. return 0;
  83. }
  84. int msgbox_cache_close(uint16_t msgbox_id, bool bsync)
  85. {
  86. return ui_msgbox_close(msgbox_id, bsync);
  87. }
  88. int msgbox_cache_paint(uint16_t msgbox_id, void * user_data)
  89. {
  90. return ui_msgbox_paint(msgbox_id, user_data);
  91. }
  92. void msgbox_cache_dump(void)
  93. {
  94. sys_dnode_t *node;
  95. os_printk("msgbox (en %d, num %d, top %u):\n", msgbox_ctx.en,
  96. msgbox_cache_num_popup_get(), msgbox_cache_get_top());
  97. if (msgbox_cache_num_popup_get() == 0) {
  98. return;
  99. }
  100. os_printk(" id | order | hwnd | view\n"
  101. "-----+-------+----------+-----\n");
  102. SYS_DLIST_FOR_EACH_NODE(&msgbox_ctx.popup_list, node) {
  103. msgbox_popup_t * popup = CONTAINER_OF(node, msgbox_popup_t, node);
  104. os_printk(" %3u | %02x | %08x | %3u\n", popup->dsc->id,
  105. popup->dsc->order, popup->hwnd, popup->view_id);
  106. }
  107. os_printk("\n");
  108. }
  109. static void _msgbox_popup(uint16_t msgbox_id, uint8_t msg_id, void * msg_data, void * user_data)
  110. {
  111. uint16_t view_id = (uint16_t)(uintptr_t)msg_data;
  112. view_data_t *view_data = view_get_data(view_id);
  113. lv_obj_t *top_layer = lv_disp_get_layer_top(view_data->display);
  114. const msgbox_dsc_t *dsc;
  115. msgbox_popup_t *popup = NULL;
  116. sys_dnode_t *node;
  117. int index;
  118. if (!top_layer) {
  119. goto fail_exit;
  120. }
  121. if (_msgbox_find_popup_by_id(msgbox_id)) {
  122. goto fail_exit;
  123. }
  124. dsc = _msgbox_find_dsc(msgbox_id);
  125. assert(dsc != NULL && (dsc->popup_cb != NULL || dsc->handler != NULL));
  126. popup = mem_malloc(sizeof(*popup));
  127. if (popup == NULL) {
  128. if (dsc->handler) {
  129. dsc->handler(msgbox_id, MSG_MSGBOX_CANCEL, user_data, NULL);
  130. } else {
  131. dsc->popup_cb(msgbox_id, MSG_MSGBOX_CANCEL, NULL, user_data);
  132. }
  133. goto fail_exit;
  134. }
  135. memset(popup, 0, sizeof(*popup));
  136. popup->dsc = dsc;
  137. popup->view_id = view_id;
  138. popup->user_data = user_data;
  139. popup->hwnd = lv_obj_create(top_layer);
  140. if (popup->hwnd == NULL) {
  141. goto fail_exit;
  142. }
  143. lv_obj_set_size(popup->hwnd, LV_PCT(100), LV_PCT(100));
  144. lv_obj_add_flag(popup->hwnd, LV_OBJ_FLAG_CLICKABLE);
  145. lv_obj_clear_flag(popup->hwnd, LV_OBJ_FLAG_GESTURE_BUBBLE);
  146. if (dsc->handler) {
  147. if (dsc->handler(msgbox_id, MSG_MSGBOX_POPUP, user_data, popup->hwnd)) {
  148. goto fail_exit;
  149. }
  150. } else {
  151. if (dsc->popup_cb(msgbox_id, MSG_MSGBOX_POPUP, popup->hwnd, user_data) == NULL) {
  152. goto fail_exit;
  153. }
  154. }
  155. index = lv_obj_get_child_cnt(top_layer) - 1;
  156. SYS_DLIST_FOR_EACH_NODE(&msgbox_ctx.popup_list, node) {
  157. msgbox_popup_t * item = CONTAINER_OF(node, msgbox_popup_t, node);
  158. if (item->dsc->order < popup->dsc->order) index--;
  159. }
  160. /* change the order */
  161. lv_obj_move_to_index(popup->hwnd, index);
  162. sys_dlist_append(&msgbox_ctx.popup_list, &popup->node);
  163. _msgbox_update_top(lv_obj_get_child(top_layer, lv_obj_get_child_cnt(top_layer) - 1));
  164. _call_view_proc(popup->view_id, popup->dsc->id, MSG_MSGBOX_POPUP);
  165. SYS_LOG_INF("create msgbox %d (%p), attached to view %u, top %p",
  166. msgbox_id, popup->hwnd, view_id, top_layer);
  167. return;
  168. fail_exit:
  169. SYS_LOG_ERR("fail to create msgbox %d", msgbox_id);
  170. if (popup) {
  171. if (popup->hwnd) {
  172. lv_obj_del(popup->hwnd);
  173. }
  174. mem_free(popup);
  175. }
  176. atomic_dec(&msgbox_ctx.num_popups);
  177. ui_gesture_unlock_scroll();
  178. }
  179. static void _msgbox_cancel(uint16_t msgbox_id, uint8_t msg_id, void * msg_data, void * user_data)
  180. {
  181. msgbox_popup_t *popup = _msgbox_find_popup_by_id(msgbox_id);
  182. SYS_LOG_WRN("msgbox %u cancel", msgbox_id);
  183. if (popup == NULL) {
  184. const msgbox_dsc_t *dsc = _msgbox_find_dsc(msgbox_id);
  185. assert(dsc != NULL);
  186. if (dsc->handler) {
  187. dsc->handler(msgbox_id, MSG_MSGBOX_CANCEL, user_data, NULL);
  188. } else {
  189. dsc->popup_cb(msgbox_id, MSG_MSGBOX_CANCEL, NULL, user_data);
  190. }
  191. atomic_dec(&msgbox_ctx.num_popups);
  192. }
  193. }
  194. static void _msgbox_paint(uint16_t msgbox_id, uint8_t msg_id, void * msg_data)
  195. {
  196. msgbox_popup_t *popup = _msgbox_find_popup_by_id(msgbox_id);
  197. if (popup) {
  198. if (popup->dsc->handler) {
  199. popup->dsc->handler(msgbox_id, MSG_MSGBOX_PAINT, msg_data, popup->hwnd);
  200. } else {
  201. popup->dsc->popup_cb(msgbox_id, MSG_MSGBOX_PAINT, msg_data, popup->user_data);
  202. }
  203. }
  204. }
  205. static void _msgbox_close(uint16_t msgbox_id, uint8_t msg_id, void * msg_data)
  206. {
  207. uint16_t view_id = (uint16_t)(uintptr_t)msg_data;
  208. sys_dnode_t *dn, *dns;
  209. SYS_DLIST_FOR_EACH_NODE_SAFE(&msgbox_ctx.popup_list, dn, dns) {
  210. msgbox_popup_t * popup = CONTAINER_OF(dn, msgbox_popup_t, node);
  211. if ((msgbox_id == popup->dsc->id) ||
  212. (msgbox_id == MSGBOX_ID_ALL && (popup->view_id == view_id || view_id == VIEW_INVALID_ID))) {
  213. lv_obj_t * top_layer = lv_obj_get_parent(popup->hwnd);
  214. SYS_LOG_INF("delete msgbox %d (%p)", popup->dsc->id, popup->hwnd);
  215. sys_dlist_remove(&popup->node);
  216. if (popup->dsc->handler) {
  217. popup->dsc->handler(popup->dsc->id, MSG_MSGBOX_CLOSE, NULL, popup->hwnd);
  218. } else {
  219. popup->dsc->popup_cb(popup->dsc->id, MSG_MSGBOX_CLOSE, popup->hwnd, popup->user_data);
  220. }
  221. if (atomic_dec(&msgbox_ctx.num_popups) == 1) {
  222. /* no msgbox now */
  223. msgbox_ctx.top = MSGBOX_ID_ALL;
  224. } else if (top_layer && popup->hwnd == lv_obj_get_child(top_layer, lv_obj_get_child_cnt(top_layer) - 1)) {
  225. _msgbox_update_top(lv_obj_get_child(top_layer, lv_obj_get_child_cnt(top_layer) - 2));
  226. }
  227. lv_obj_del(popup->hwnd);
  228. _call_view_proc(popup->view_id, popup->dsc->id, MSG_MSGBOX_CLOSE);
  229. mem_free(popup);
  230. ui_gesture_unlock_scroll();
  231. }
  232. }
  233. }
  234. static void _msgbox_proc_key(ui_key_msg_data_t *data)
  235. {
  236. if (msgbox_ctx.top != MSGBOX_INVALID_ID) {
  237. msgbox_popup_t *popup = _msgbox_find_popup_by_id(msgbox_ctx.top);
  238. if (popup) {
  239. if (popup->dsc->handler) {
  240. popup->dsc->handler(popup->dsc->id, MSG_MSGBOX_KEY, data, popup->hwnd);
  241. } else {
  242. popup->dsc->popup_cb(popup->dsc->id, MSG_MSGBOX_KEY, data, popup->user_data);
  243. }
  244. if (data->done)
  245. SYS_LOG_INF("msgbox %u done key 0x%x", popup->dsc->id, data->event);
  246. }
  247. }
  248. }
  249. static void _msgbox_popup_handler(uint16_t msgbox_id, uint8_t msg_id, void * msg_data, void * user_data)
  250. {
  251. switch (msg_id) {
  252. case MSG_MSGBOX_POPUP:
  253. _msgbox_popup(msgbox_id, msg_id, msg_data, user_data);
  254. break;
  255. case MSG_MSGBOX_CANCEL:
  256. _msgbox_cancel(msgbox_id, msg_id, NULL, user_data);
  257. break;
  258. case MSG_MSGBOX_CLOSE:
  259. _msgbox_close(msgbox_id, msg_id, msg_data);
  260. break;
  261. case MSG_MSGBOX_PAINT:
  262. _msgbox_paint(msgbox_id, msg_id, msg_data);
  263. break;
  264. case MSG_MSGBOX_KEY:
  265. _msgbox_proc_key(msg_data);
  266. break;
  267. default:
  268. break;
  269. }
  270. }
  271. static const msgbox_dsc_t *_msgbox_find_dsc(uint16_t msgbox_id)
  272. {
  273. int i;
  274. for (i = msgbox_ctx.num - 1; i >= 0; i--) {
  275. if (msgbox_id == msgbox_ctx.dsc[i].id) {
  276. return &msgbox_ctx.dsc[i];
  277. }
  278. }
  279. return NULL;
  280. }
  281. static msgbox_popup_t *_msgbox_find_popup_by_id(uint16_t msgbox_id)
  282. {
  283. sys_dnode_t *node;
  284. SYS_DLIST_FOR_EACH_NODE(&msgbox_ctx.popup_list, node) {
  285. msgbox_popup_t * item = CONTAINER_OF(node, msgbox_popup_t, node);
  286. if (item->dsc->id == msgbox_id) {
  287. return item;
  288. }
  289. }
  290. return NULL;
  291. }
  292. static msgbox_popup_t *_msgbox_find_popup_by_hwnd(void * hwnd)
  293. {
  294. sys_dnode_t *node;
  295. SYS_DLIST_FOR_EACH_NODE(&msgbox_ctx.popup_list, node) {
  296. msgbox_popup_t *item = CONTAINER_OF(node, msgbox_popup_t, node);
  297. if (item->hwnd == hwnd) {
  298. return item;
  299. }
  300. }
  301. return NULL;
  302. }
  303. static void _msgbox_update_top(void * hwnd)
  304. {
  305. msgbox_popup_t *popup;
  306. popup = _msgbox_find_popup_by_hwnd(hwnd);
  307. if (popup) {
  308. msgbox_ctx.top = popup->dsc->id;
  309. } else {
  310. msgbox_ctx.top = MSGBOX_INVALID_ID;
  311. }
  312. }
  313. static void _call_view_proc(uint16_t view_id, uint16_t msgbox_id, uint8_t msg_id)
  314. {
  315. view_entry_t *entry = view_manager_get_view_entry(view_id);
  316. if (entry != NULL) {
  317. if (entry->proc2) {
  318. entry->proc2(view_id, view_get_data(view_id), msg_id, (void *)(uintptr_t)msgbox_id);
  319. } else if (entry->proc) {
  320. entry->proc(view_id, msg_id, (void *)(uintptr_t)msgbox_id);
  321. }
  322. }
  323. }