view_cache.c 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939
  1. /*
  2. * Copyright (c) 2019 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file view cache interface
  8. */
  9. #ifdef CONFIG_VIEW_SCROLL_MEM_DEFAULT
  10. #define LOG_MODULE_CUSTOMER
  11. #include <os_common_api.h>
  12. #include <ui_manager.h>
  13. #include <view_cache.h>
  14. #include <string.h>
  15. #include <assert.h>
  16. LOG_MODULE_REGISTER(view_cache, LOG_LEVEL_INF);
  17. #define MAX_VIEW_CACHE (sizeof(uint32_t) * 8)
  18. typedef struct {
  19. const view_cache_dsc_t *dsc;
  20. uint32_t stat;
  21. uint8_t rotate : 1;
  22. uint8_t shrunk : 1;
  23. int8_t main_idx; /* focused main idx in vlist */
  24. int8_t focus_idx; /* focused idx also considering cross_vlist */
  25. int8_t load_idx; /* the index that is loading */
  26. /* save the initial param */
  27. int8_t init_main_idx;
  28. int8_t init_focus_idx;
  29. } view_cache_ctx_t;
  30. static int view_cache_view_transforming = 0;
  31. static int view_cache_serial_load_paused = 0;
  32. static int _view_cache_set_focus(uint16_t view_id);
  33. static int _view_cache_set_focus_idx(int8_t main_idx, bool in_restore);
  34. static int _view_cache_shrink(void);
  35. static void _view_cache_restore(void);
  36. static int8_t _view_cache_rotate_main_idx(int8_t idx);
  37. static view_cache_ctx_t view_cache_ctx;
  38. static view_cache_focus_cb_t last_focus_cb;
  39. static uint16_t last_focus_view;
  40. static OS_MUTEX_DEFINE(view_cache_mutex);
  41. static uint16_t _view_cache_get_view_id(uint8_t idx)
  42. {
  43. if (idx < view_cache_ctx.dsc->num) {
  44. return view_cache_ctx.dsc->vlist[idx];
  45. } else if (idx < view_cache_ctx.dsc->num + 2) {
  46. return view_cache_ctx.dsc->cross_vlist[idx - view_cache_ctx.dsc->num];
  47. } else {
  48. return VIEW_INVALID_ID;
  49. }
  50. }
  51. static const void * _view_cache_get_presenter(uint8_t idx)
  52. {
  53. if (idx < view_cache_ctx.dsc->num) {
  54. return view_cache_ctx.dsc->plist ? view_cache_ctx.dsc->plist[idx] : NULL;
  55. } else if (idx < view_cache_ctx.dsc->num + 2) {
  56. return view_cache_ctx.dsc->cross_plist[idx - view_cache_ctx.dsc->num];
  57. } else {
  58. return NULL;
  59. }
  60. }
  61. static uint8_t _view_cache_get_create_flags(uint8_t idx)
  62. {
  63. if (idx < view_cache_ctx.dsc->num && view_cache_ctx.dsc->vlist_create_flags) {
  64. return view_cache_ctx.dsc->vlist_create_flags[idx];
  65. } else {
  66. return 0;
  67. }
  68. }
  69. static int8_t _view_cache_get_main_idx(uint16_t view_id)
  70. {
  71. int8_t idx;
  72. for (idx = view_cache_ctx.dsc->num - 1; idx >= 0; idx--) {
  73. if (view_cache_ctx.dsc->vlist[idx] == view_id) {
  74. return idx;
  75. }
  76. }
  77. return -1;
  78. }
  79. static int8_t _view_cache_get_idx(uint16_t view_id)
  80. {
  81. if (view_id == view_cache_ctx.dsc->cross_vlist[0]) {
  82. return view_cache_ctx.dsc->num;
  83. } else if (view_id == view_cache_ctx.dsc->cross_vlist[1]) {
  84. return view_cache_ctx.dsc->num + 1;
  85. } else {
  86. return _view_cache_get_main_idx(view_id);
  87. }
  88. }
  89. static int _view_cache_load(uint8_t idx, uint8_t create_flags)
  90. {
  91. uint16_t view_id;
  92. if (view_cache_ctx.stat & (1 << idx))
  93. return -EALREADY;
  94. view_id = _view_cache_get_view_id(idx);
  95. if (view_id != VIEW_INVALID_ID) {
  96. view_cache_ctx.stat |= (1 << idx);
  97. create_flags |= _view_cache_get_create_flags(idx);
  98. return ui_view_create(view_id, _view_cache_get_presenter(idx), create_flags);
  99. }
  100. return -EINVAL;
  101. }
  102. static void _view_cache_unload(uint8_t idx)
  103. {
  104. uint16_t view_id;
  105. if (!(view_cache_ctx.stat & (1 << idx)))
  106. return;
  107. view_id = _view_cache_get_view_id(idx);
  108. assert(view_id != VIEW_INVALID_ID);
  109. view_cache_ctx.stat &= ~(1 << idx);
  110. ui_view_delete(view_id);
  111. }
  112. static int _view_cache_set_attr(uint8_t idx, uint8_t attr, bool keep_pos, bool by_msg)
  113. {
  114. uint16_t view_id;
  115. if (!(view_cache_ctx.stat & (1 << idx)))
  116. return -EINVAL;
  117. view_id = _view_cache_get_view_id(idx);
  118. assert(view_id != VIEW_INVALID_ID);
  119. if (by_msg) {
  120. return ui_view_set_drag_attribute(view_id, attr, keep_pos);
  121. } else {
  122. return view_set_drag_attribute(view_id, attr, keep_pos);
  123. }
  124. }
  125. static int8_t _view_cache_rotate_main_idx(int8_t idx)
  126. {
  127. while (idx < 0)
  128. idx += view_cache_ctx.dsc->num;
  129. while (idx >= view_cache_ctx.dsc->num)
  130. idx -= view_cache_ctx.dsc->num;
  131. return idx;
  132. }
  133. static bool _view_cache_main_idx_is_in_range(int8_t idx)
  134. {
  135. int8_t min_idx = view_cache_ctx.main_idx - CONFIG_VIEW_CACHE_LEVEL;
  136. int8_t max_idx = view_cache_ctx.main_idx + CONFIG_VIEW_CACHE_LEVEL;
  137. if (view_cache_ctx.rotate) {
  138. if (view_cache_ctx.dsc->num <= (CONFIG_VIEW_CACHE_LEVEL * 2 + 1))
  139. return true;
  140. idx = _view_cache_rotate_main_idx(idx);
  141. if (min_idx < 0) {
  142. min_idx += view_cache_ctx.dsc->num;
  143. if (idx >= min_idx || idx <= max_idx)
  144. return true;
  145. } else if (max_idx >= view_cache_ctx.dsc->num) {
  146. max_idx -= view_cache_ctx.dsc->num;
  147. if (idx >= min_idx || idx <= max_idx)
  148. return true;
  149. } else if (idx >= min_idx && idx <= max_idx) {
  150. return true;
  151. }
  152. } else {
  153. if (idx >= UI_MAX(min_idx, 0) &&
  154. idx <= UI_MIN(max_idx, view_cache_ctx.dsc->num - 1))
  155. return true;
  156. }
  157. return false;
  158. }
  159. static int _view_cache_load_main(int8_t idx, uint8_t create_flags)
  160. {
  161. if (view_cache_ctx.rotate) {
  162. idx = _view_cache_rotate_main_idx(idx);
  163. } else if (idx < 0 || idx >= view_cache_ctx.dsc->num) {
  164. return -EINVAL;
  165. }
  166. return _view_cache_load(idx, create_flags);
  167. }
  168. static void _view_cache_unload_main(int8_t idx, bool forced)
  169. {
  170. if (view_cache_ctx.rotate) {
  171. if (!forced && _view_cache_main_idx_is_in_range(idx))
  172. return;
  173. idx = _view_cache_rotate_main_idx(idx);
  174. } else if (idx < 0 || idx >= view_cache_ctx.dsc->num) {
  175. return;
  176. }
  177. _view_cache_unload(idx);
  178. }
  179. static int _view_cache_set_attr_main(int8_t idx, uint8_t attr, bool by_msg)
  180. {
  181. if (view_cache_ctx.rotate) {
  182. idx = _view_cache_rotate_main_idx(idx);
  183. } else if (idx < 0 || idx >= view_cache_ctx.dsc->num) {
  184. return -EINVAL;
  185. }
  186. return _view_cache_set_attr(idx, attr, false, by_msg);
  187. }
  188. static uint8_t _view_cache_rotate_drag_attr(uint8_t drag_attr)
  189. {
  190. static const uint8_t s_attr_rotated[][4] = {
  191. { UI_DRAG_DROPLEFT, UI_DRAG_DROPRIGHT, UI_DRAG_DROPUP, UI_DRAG_DROPDOWN },
  192. { UI_DRAG_DROPUP, UI_DRAG_DROPDOWN, UI_DRAG_DROPRIGHT, UI_DRAG_DROPLEFT },
  193. { UI_DRAG_DROPRIGHT, UI_DRAG_DROPLEFT, UI_DRAG_DROPDOWN, UI_DRAG_DROPUP },
  194. };
  195. uint16_t rotation = view_manager_get_disp_rotation();
  196. const uint8_t *rotated_attr;
  197. uint8_t new_attr = 0;
  198. if (drag_attr == 0 || rotation == 0)
  199. return drag_attr;
  200. if (rotation == 90) {
  201. rotated_attr = s_attr_rotated[0];
  202. } else if (rotation == 180) {
  203. rotated_attr = s_attr_rotated[1];
  204. } else {
  205. rotated_attr = s_attr_rotated[2];
  206. }
  207. if (drag_attr & 0x0f) {
  208. for (int i = 0; i < 4; i++) {
  209. if (drag_attr & (1 << i)) {
  210. new_attr |= rotated_attr[i];
  211. }
  212. }
  213. }
  214. if (drag_attr & 0xf0) {
  215. for (int i = 4; i < 8; i++) {
  216. if (drag_attr & (1 << i)) {
  217. new_attr |= rotated_attr[i - 4] << 4;
  218. }
  219. }
  220. }
  221. return new_attr;
  222. }
  223. static uint8_t _view_cache_decide_attr_main(int8_t idx)
  224. {
  225. int8_t left_idx = view_cache_ctx.main_idx - 1;
  226. int8_t right_idx = view_cache_ctx.main_idx + 1;
  227. uint8_t attr = 0;
  228. if (view_cache_ctx.rotate) {
  229. idx = _view_cache_rotate_main_idx(idx);
  230. left_idx = _view_cache_rotate_main_idx(left_idx);
  231. right_idx = _view_cache_rotate_main_idx(right_idx);
  232. }
  233. if (view_cache_ctx.dsc->type == LANDSCAPE) {
  234. if (view_cache_ctx.rotate && view_cache_ctx.dsc->num == 2) {
  235. attr = (idx == view_cache_ctx.main_idx) ? 0 :
  236. (UI_DRAG_MOVERIGHT | UI_DRAG_MOVELEFT);
  237. }
  238. if (idx == left_idx)
  239. attr = UI_DRAG_MOVERIGHT;
  240. else if (idx == right_idx)
  241. attr = UI_DRAG_MOVELEFT;
  242. } else {
  243. if (view_cache_ctx.rotate && view_cache_ctx.dsc->num == 2) {
  244. attr = (idx == view_cache_ctx.main_idx) ? 0 :
  245. (UI_DRAG_MOVEDOWN | UI_DRAG_MOVEUP);
  246. }
  247. if (idx == left_idx)
  248. attr = UI_DRAG_MOVEDOWN;
  249. else if (idx == right_idx)
  250. attr = UI_DRAG_MOVEUP;
  251. }
  252. return _view_cache_rotate_drag_attr(attr);
  253. }
  254. static uint8_t _view_cache_decide_attr_cross(int8_t idx)
  255. {
  256. uint8_t attr = 0;
  257. if (view_cache_ctx.dsc->type == LANDSCAPE) {
  258. if (idx == view_cache_ctx.dsc->num)
  259. attr = UI_DRAG_DROPDOWN;
  260. else if (idx == view_cache_ctx.dsc->num + 1)
  261. attr = UI_DRAG_DROPUP;
  262. } else {
  263. if (idx == view_cache_ctx.dsc->num)
  264. attr = UI_DRAG_DROPRIGHT;
  265. else if (idx == view_cache_ctx.dsc->num + 1)
  266. attr = UI_DRAG_DROPLEFT;
  267. }
  268. return _view_cache_rotate_drag_attr(attr);
  269. }
  270. static void _view_cache_serial_load(void)
  271. {
  272. int8_t main_idx = view_cache_ctx.main_idx;
  273. int8_t load_idx;
  274. /* load the "cross_attached_view" main view as soon as possible */
  275. if (!_view_cache_load(main_idx, UI_CREATE_FLAG_SHOW)) {
  276. return;
  277. }
  278. /* load cross views */
  279. for(; view_cache_ctx.load_idx >= view_cache_ctx.dsc->num; view_cache_ctx.load_idx--) {
  280. if (!_view_cache_load(view_cache_ctx.load_idx, 0)) {
  281. if (view_cache_ctx.dsc->cross_attached_view == VIEW_INVALID_ID ||
  282. view_cache_ctx.dsc->cross_attached_view == view_cache_ctx.dsc->vlist[main_idx]) {
  283. _view_cache_set_attr(view_cache_ctx.load_idx,
  284. _view_cache_decide_attr_cross(view_cache_ctx.load_idx), false, true);
  285. }
  286. return;
  287. }
  288. }
  289. /* load left main view */
  290. if (view_cache_ctx.rotate) {
  291. load_idx = _view_cache_rotate_main_idx(main_idx - 1);
  292. } else {
  293. load_idx = main_idx - 1;
  294. }
  295. if (_view_cache_main_idx_is_in_range(load_idx) &&
  296. !_view_cache_load(load_idx, 0)) {
  297. _view_cache_set_attr_main(load_idx, _view_cache_decide_attr_main(main_idx - 1), true);
  298. return;
  299. }
  300. /* load right main view */
  301. if (view_cache_ctx.rotate) {
  302. load_idx = _view_cache_rotate_main_idx(main_idx + 1);
  303. } else {
  304. load_idx = main_idx + 1;
  305. }
  306. if (_view_cache_main_idx_is_in_range(load_idx) &&
  307. !_view_cache_load(load_idx, 0)) {
  308. _view_cache_set_attr_main(load_idx, _view_cache_decide_attr_main(main_idx + 1), true);
  309. return;
  310. }
  311. /* load other main views */
  312. for (; view_cache_ctx.load_idx >= 0; view_cache_ctx.load_idx--) {
  313. if (_view_cache_main_idx_is_in_range(view_cache_ctx.load_idx) &&
  314. !_view_cache_load(view_cache_ctx.load_idx, 0)) {
  315. return;
  316. }
  317. }
  318. view_cache_ctx.load_idx = -1;
  319. if (view_cache_ctx.dsc->event_cb) {
  320. view_cache_ctx.dsc->event_cb(VIEW_CACHE_EVT_LOAD_END);
  321. }
  322. }
  323. static void _view_cache_scroll_cb(uint16_t view_id, uint8_t msg_id)
  324. {
  325. if (msg_id != MSG_VIEW_SCROLL_END) {
  326. return;
  327. }
  328. os_mutex_lock(&view_cache_mutex, OS_FOREVER);
  329. if (view_cache_ctx.dsc == NULL) {
  330. goto out_unlock;
  331. }
  332. _view_cache_set_focus(view_id);
  333. out_unlock:
  334. os_mutex_unlock(&view_cache_mutex);
  335. }
  336. static void _view_cache_monitor_cb(uint16_t view_id, uint8_t msg_id, void *msg_data)
  337. {
  338. int8_t idx;
  339. os_mutex_lock(&view_cache_mutex, OS_FOREVER);
  340. if (view_cache_ctx.dsc == NULL) {
  341. if (msg_id == MSG_VIEW_DEFOCUS && last_focus_cb && view_id == last_focus_view) {
  342. last_focus_cb(view_id, false);
  343. last_focus_cb = NULL;
  344. }
  345. goto out_unlock;
  346. }
  347. idx = _view_cache_get_idx(view_id);
  348. if (idx < 0) {
  349. //if (msg_id == MSG_VIEW_FOCUS && view_cache_ctx.shrunk == 0) {
  350. // _view_cache_shrink();
  351. //}
  352. goto out_unlock;
  353. }
  354. if (msg_id == MSG_VIEW_FOCUS || msg_id == MSG_VIEW_DEFOCUS) {
  355. bool focused = (msg_id == MSG_VIEW_FOCUS);
  356. if (focused) {
  357. view_cache_ctx.focus_idx = idx;
  358. if (view_cache_ctx.shrunk)
  359. _view_cache_restore();
  360. }
  361. if (view_cache_ctx.dsc->focus_cb)
  362. view_cache_ctx.dsc->focus_cb(view_id, focused);
  363. } else if (msg_id == MSG_VIEW_SCROLL_BEGIN || msg_id == MSG_VIEW_SCROLL_END) {
  364. if (view_cache_ctx.dsc->monitor_cb)
  365. view_cache_ctx.dsc->monitor_cb(view_id, msg_id);
  366. } else if (msg_id == MSG_VIEW_LAYOUT) {
  367. if (view_cache_ctx.load_idx >= 0) {
  368. if (view_cache_view_transforming) {
  369. SYS_LOG_INF("view serial load paused after %d\n", view_id);
  370. view_cache_serial_load_paused = 1;
  371. } else {
  372. _view_cache_serial_load();
  373. }
  374. } else {
  375. int8_t idx = _view_cache_get_main_idx(view_id);
  376. if (idx >= 0) { /* in case that previous set_attr() failed */
  377. _view_cache_set_attr_main(idx, _view_cache_decide_attr_main(idx), false);
  378. }
  379. }
  380. } else if(msg_id == MSG_VIEW_TRANSFORM_START) {
  381. view_cache_view_transforming = 1;
  382. } else if (msg_id == MSG_VIEW_TRANSFORM_END) {
  383. SYS_LOG_INF("view serial load resume after %d\n", view_id);
  384. view_cache_view_transforming = 0;
  385. if (view_cache_serial_load_paused == 1) {
  386. view_cache_serial_load_paused = 0;
  387. _view_cache_serial_load();
  388. }
  389. }
  390. out_unlock:
  391. os_mutex_unlock(&view_cache_mutex);
  392. }
  393. int view_cache_init(const view_cache_dsc_t *dsc, uint16_t init_view)
  394. {
  395. return view_cache_init2(dsc, init_view, VIEW_INVALID_ID);
  396. }
  397. int view_cache_init2(const view_cache_dsc_t *dsc,
  398. uint16_t init_focus_view, uint16_t init_main_view)
  399. {
  400. int8_t init_main_idx = -1; /* index of init_main_view */
  401. int8_t main_idx = -1;
  402. int8_t cross_idx = -1;
  403. int8_t cross_attached_idx = -1;
  404. int8_t idx;
  405. int res = -EINVAL;
  406. /* Also consider the cross views */
  407. if (dsc == NULL || dsc->num <= 0 || dsc->num + 2 > MAX_VIEW_CACHE) {
  408. goto out_fail;
  409. }
  410. if (init_focus_view == VIEW_INVALID_ID) {
  411. goto out_fail;
  412. }
  413. for (idx = dsc->num - 1; idx >= 0; idx--) {
  414. if (dsc->vlist[idx] == VIEW_INVALID_ID)
  415. goto out_fail;
  416. if (dsc->vlist[idx] == init_focus_view)
  417. main_idx = idx;
  418. if (dsc->vlist[idx] == init_main_view)
  419. init_main_idx = idx;
  420. if (dsc->vlist[idx] == dsc->cross_attached_view)
  421. cross_attached_idx = idx;
  422. }
  423. if (main_idx < 0) {
  424. if (init_focus_view == dsc->cross_vlist[0]) {
  425. cross_idx = dsc->num;
  426. } else if (init_focus_view == dsc->cross_vlist[1]) {
  427. cross_idx = dsc->num + 1;
  428. } else {
  429. goto out_fail;
  430. }
  431. if (init_main_idx >= 0) {
  432. main_idx = init_main_idx;
  433. } else if (cross_attached_idx >= 0) {
  434. main_idx = cross_attached_idx;
  435. } else { /* FIXME: just select the middle view ? */
  436. main_idx = (dsc->num / 2);
  437. }
  438. }
  439. os_mutex_lock(&view_cache_mutex, OS_FOREVER);
  440. if (view_cache_ctx.dsc) {
  441. SYS_LOG_WRN("view_cache: already init");
  442. res = -EALREADY;
  443. goto out_unlock;
  444. }
  445. view_cache_ctx.dsc = dsc;
  446. view_cache_ctx.main_idx = -1;
  447. view_cache_ctx.focus_idx = -1;
  448. view_cache_ctx.rotate = (view_cache_ctx.dsc->rotate && view_cache_ctx.dsc->num >= 2);
  449. view_cache_ctx.init_main_idx = main_idx;
  450. view_cache_ctx.init_focus_idx = (cross_idx >= 0) ? cross_idx : main_idx;
  451. ui_manager_set_scroll_callback(_view_cache_scroll_cb);
  452. ui_manager_set_monitor_callback(_view_cache_monitor_cb);
  453. if (dsc->serial_load == 0) {
  454. /* mark serial loading unnecessary*/
  455. view_cache_ctx.load_idx = -1;
  456. if (cross_idx < 0) {
  457. /* load focused view */
  458. _view_cache_load(main_idx, UI_CREATE_FLAG_SHOW);
  459. }
  460. /* load cross views */
  461. for (idx = dsc->num; idx <= dsc->num + 1; idx++) {
  462. uint8_t create_flag = 0;
  463. bool keep_pos = false;
  464. if (cross_idx >= 0 && cross_idx == idx) {
  465. create_flag = UI_CREATE_FLAG_SHOW;
  466. keep_pos = true;
  467. }
  468. if (!_view_cache_load(idx, create_flag)) {
  469. _view_cache_set_attr(idx, _view_cache_decide_attr_cross(idx), keep_pos, true);
  470. }
  471. }
  472. /* load other main views */
  473. _view_cache_set_focus(dsc->vlist[main_idx]);
  474. } else {
  475. view_cache_ctx.load_idx = dsc->num + 1;
  476. view_cache_ctx.main_idx = main_idx;
  477. if (dsc->event_cb) {
  478. dsc->event_cb(VIEW_CACHE_EVT_LOAD_BEGIN);
  479. }
  480. /* load focused view */
  481. if (cross_idx < 0) {
  482. _view_cache_load(main_idx, UI_CREATE_FLAG_SHOW);
  483. } else if (!_view_cache_load(cross_idx, UI_CREATE_FLAG_SHOW)) {
  484. _view_cache_set_attr(cross_idx, _view_cache_decide_attr_cross(cross_idx), true, true);
  485. }
  486. }
  487. res = 0;
  488. out_unlock:
  489. SYS_LOG_INF("view_cache: init %u, main %u (res=%d)", init_focus_view, init_main_view, res);
  490. os_mutex_unlock(&view_cache_mutex);
  491. return res;
  492. out_fail:
  493. os_mutex_lock(&view_cache_mutex, OS_FOREVER);
  494. goto out_unlock;
  495. }
  496. int view_cache_deinit(void)
  497. {
  498. int8_t idx;
  499. os_mutex_lock(&view_cache_mutex, OS_FOREVER);
  500. if (view_cache_ctx.dsc == NULL) {
  501. SYS_LOG_WRN("view_cache: not init yet");
  502. os_mutex_unlock(&view_cache_mutex);
  503. return EALREADY;
  504. }
  505. ui_manager_set_scroll_callback(NULL);
  506. if (view_cache_ctx.dsc->serial_load && view_cache_ctx.load_idx >= 0 &&
  507. view_cache_ctx.dsc->event_cb) {
  508. view_cache_ctx.dsc->event_cb(VIEW_CACHE_EVT_LOAD_CANCEL);
  509. }
  510. if (view_cache_ctx.focus_idx >= 0 && view_cache_ctx.dsc->focus_cb) {
  511. last_focus_view = _view_cache_get_view_id(view_cache_ctx.focus_idx);
  512. last_focus_cb = view_cache_ctx.dsc->focus_cb;
  513. } else {
  514. last_focus_cb = NULL;
  515. }
  516. /* delete the main views first to avoid unexpected focus changes */
  517. for (idx = 0; idx < view_cache_ctx.dsc->num + 2; idx++) {
  518. _view_cache_unload(idx);
  519. }
  520. ui_manager_set_monitor_callback(NULL);
  521. memset(&view_cache_ctx, 0, sizeof(view_cache_ctx));
  522. if (is_in_ui_thread() && last_focus_cb) { /* notify the defocus */
  523. last_focus_cb(last_focus_view, false);
  524. last_focus_cb = NULL;
  525. }
  526. view_cache_view_transforming = 0;
  527. view_cache_serial_load_paused = 0;
  528. SYS_LOG_INF("view_cache: deinit");
  529. os_mutex_unlock(&view_cache_mutex);
  530. return 0;
  531. }
  532. void view_cache_wait_last_focus_cb(void)
  533. {
  534. /* neither in uisrv nor executed yet */
  535. if (last_focus_cb) {
  536. ui_message_wait_reply();
  537. }
  538. }
  539. uint16_t view_cache_get_focus_view(void)
  540. {
  541. uint16_t view_id = VIEW_INVALID_ID;
  542. os_mutex_lock(&view_cache_mutex, OS_FOREVER);
  543. if (view_cache_ctx.dsc) {
  544. int8_t focus_idx = (view_cache_ctx.focus_idx >= 0) ?
  545. view_cache_ctx.focus_idx : view_cache_ctx.init_focus_idx;
  546. view_id = _view_cache_get_view_id(focus_idx);
  547. }
  548. os_mutex_unlock(&view_cache_mutex);
  549. return view_id;
  550. }
  551. uint16_t view_cache_get_focus_main_view(void)
  552. {
  553. uint16_t view_id = VIEW_INVALID_ID;
  554. os_mutex_lock(&view_cache_mutex, OS_FOREVER);
  555. if (view_cache_ctx.dsc) {
  556. int8_t main_idx = (view_cache_ctx.main_idx >= 0) ?
  557. view_cache_ctx.main_idx : view_cache_ctx.init_main_idx;
  558. view_id = _view_cache_get_view_id(main_idx);
  559. }
  560. os_mutex_unlock(&view_cache_mutex);
  561. return view_id;
  562. }
  563. int view_cache_set_focus_view(uint16_t view_id)
  564. {
  565. const view_cache_dsc_t *dsc;
  566. uint16_t main_view_Id;
  567. int8_t idx;
  568. SYS_LOG_INF("view_cache: set focus view %u", view_id);
  569. os_mutex_lock(&view_cache_mutex, OS_FOREVER);
  570. dsc = view_cache_ctx.dsc;
  571. if (dsc == NULL) {
  572. os_mutex_unlock(&view_cache_mutex);
  573. return -ESRCH;
  574. }
  575. idx = _view_cache_get_idx(view_id);
  576. if (idx < 0) {
  577. os_mutex_unlock(&view_cache_mutex);
  578. return -EINVAL;
  579. }
  580. int8_t focus_idx = (view_cache_ctx.focus_idx >= 0) ?
  581. view_cache_ctx.focus_idx : view_cache_ctx.init_focus_idx;
  582. if (idx == focus_idx) {
  583. os_mutex_unlock(&view_cache_mutex);
  584. return 0;
  585. }
  586. main_view_Id = _view_cache_get_view_id(view_cache_ctx.main_idx);
  587. os_mutex_unlock(&view_cache_mutex);
  588. view_cache_deinit();
  589. view_cache_wait_last_focus_cb();
  590. return view_cache_init2(dsc, view_id, main_view_Id);
  591. }
  592. int view_cache_shrink(void)
  593. {
  594. int res = 0;
  595. os_mutex_lock(&view_cache_mutex, OS_FOREVER);
  596. if (view_cache_ctx.dsc == NULL) {
  597. res = -ESRCH;
  598. goto out_unlock;
  599. }
  600. if (view_cache_ctx.shrunk) {
  601. res = -EALREADY;
  602. goto out_unlock;
  603. }
  604. res = _view_cache_shrink();
  605. out_unlock:
  606. os_mutex_unlock(&view_cache_mutex);
  607. return res;
  608. }
  609. void view_cache_dump(void)
  610. {
  611. uint8_t i;
  612. if (view_cache_ctx.dsc == NULL)
  613. return;
  614. os_printk("view cache:\n\t main(%u):", view_cache_ctx.dsc->num);
  615. for (i = 0; i < view_cache_ctx.dsc->num; i++) {
  616. os_printk(" %c%u", (i == view_cache_ctx.main_idx) ? '*' : ' ',
  617. view_cache_ctx.dsc->vlist[i]);
  618. }
  619. os_printk("\n\t cross:");
  620. for (; i < view_cache_ctx.dsc->num + 2; i++) {
  621. os_printk(" %c%u", (i == view_cache_ctx.focus_idx) ? '*' : ' ',
  622. view_cache_ctx.dsc->cross_vlist[i - view_cache_ctx.dsc->num]);
  623. }
  624. os_printk("\n\n");
  625. }
  626. static int _view_cache_set_focus(uint16_t view_id)
  627. {
  628. int8_t main_idx = _view_cache_get_main_idx(view_id);
  629. if (main_idx < 0)
  630. return -EINVAL;
  631. if (main_idx == view_cache_ctx.main_idx)
  632. return 0;
  633. return _view_cache_set_focus_idx(main_idx, false);
  634. }
  635. static int _view_cache_set_focus_idx(int8_t main_idx, bool in_restore)
  636. {
  637. uint16_t view_id = view_cache_ctx.dsc->vlist[main_idx];
  638. int8_t prefocus_idx = view_cache_ctx.main_idx;
  639. bool rotate2 = (view_cache_ctx.rotate && view_cache_ctx.dsc->num == 2);
  640. // save cur view, _view_cache_decide_attr() depends on the main_idx.
  641. view_cache_ctx.main_idx = main_idx;
  642. if (view_cache_ctx.dsc->cross_attached_view != VIEW_INVALID_ID) {
  643. uint8_t cross_attr[2] = { 0, 0 };
  644. if (view_id == view_cache_ctx.dsc->cross_attached_view) {
  645. cross_attr[0] = _view_cache_decide_attr_cross(view_cache_ctx.dsc->num);
  646. cross_attr[1] = _view_cache_decide_attr_cross(view_cache_ctx.dsc->num + 1);
  647. }
  648. _view_cache_set_attr(view_cache_ctx.dsc->num, cross_attr[0], in_restore, false);
  649. _view_cache_set_attr(view_cache_ctx.dsc->num + 1, cross_attr[1], in_restore, false);
  650. }
  651. // clear attr for old prev/next view
  652. _view_cache_set_attr_main(prefocus_idx - 1, 0, false);
  653. if (!rotate2)
  654. _view_cache_set_attr_main(prefocus_idx + 1, 0, false);
  655. // unload unused view
  656. _view_cache_unload_main(main_idx - (CONFIG_VIEW_CACHE_LEVEL + 1), false);
  657. _view_cache_unload_main(main_idx + (CONFIG_VIEW_CACHE_LEVEL + 1), false);
  658. // preload and show new cur view
  659. if (view_cache_ctx.load_idx >= 0) {
  660. view_cache_ctx.load_idx = -1;
  661. if (view_cache_ctx.stat & (1 << main_idx)) {
  662. _view_cache_set_attr_main(main_idx, _view_cache_decide_attr_main(main_idx), false);
  663. } else {
  664. _view_cache_load_main(main_idx, UI_CREATE_FLAG_SHOW);
  665. }
  666. if (view_cache_ctx.stat & (1 << (main_idx-1))) {
  667. _view_cache_set_attr_main(main_idx - 1, _view_cache_decide_attr_main(main_idx - 1), false);
  668. } else {
  669. _view_cache_load_main(main_idx - 1, 0);
  670. _view_cache_set_attr_main(main_idx - 1, _view_cache_decide_attr_main(main_idx - 1), false);
  671. }
  672. if (view_cache_ctx.stat & (1 << (main_idx+1))) {
  673. if (!rotate2) {
  674. _view_cache_set_attr_main(main_idx + 1, _view_cache_decide_attr_main(main_idx + 1), false);
  675. }
  676. } else {
  677. if (!rotate2) {
  678. _view_cache_load_main(main_idx + 1, 0);
  679. _view_cache_set_attr_main(main_idx + 1, _view_cache_decide_attr_main(main_idx + 1), false);
  680. }
  681. }
  682. } else {
  683. _view_cache_load_main(main_idx, UI_CREATE_FLAG_SHOW);
  684. // preload and set attr for new prev/next view
  685. _view_cache_load_main(main_idx - 1, 0);
  686. _view_cache_set_attr_main(main_idx - 1, _view_cache_decide_attr_main(main_idx - 1), false);
  687. if (!rotate2) {
  688. _view_cache_load_main(main_idx + 1, 0);
  689. _view_cache_set_attr_main(main_idx + 1, _view_cache_decide_attr_main(main_idx + 1), false);
  690. }
  691. }
  692. #if CONFIG_VIEW_CACHE_LEVEL > 1
  693. _view_cache_load_main(main_idx - CONFIG_VIEW_CACHE_LEVEL, 0);
  694. _view_cache_load_main(main_idx + CONFIG_VIEW_CACHE_LEVEL, 0);
  695. #endif
  696. return 0;
  697. }
  698. static int _view_cache_shrink(void)
  699. {
  700. int8_t main_idx = view_cache_ctx.main_idx;
  701. bool rotate2 = (view_cache_ctx.rotate && view_cache_ctx.dsc->num == 2);
  702. if (view_cache_ctx.load_idx >= 0) {
  703. SYS_LOG_WRN("serial load in progress");
  704. return -EBUSY;
  705. }
  706. SYS_LOG_INF("view_cache: shrink");
  707. #if CONFIG_VIEW_CACHE_LEVEL > 1
  708. if (view_cache_ctx.dsc->num > 3) {
  709. /* shrink the views */
  710. _view_cache_unload_main(main_idx - CONFIG_VIEW_CACHE_LEVEL, true);
  711. _view_cache_unload_main(main_idx + CONFIG_VIEW_CACHE_LEVEL, true);
  712. }
  713. #endif
  714. /* unset the drag attr */
  715. _view_cache_set_attr_main(main_idx - 1, 0, false);
  716. if (!rotate2)
  717. _view_cache_set_attr_main(main_idx + 1, 0, false);
  718. _view_cache_set_attr(view_cache_ctx.dsc->num, 0, true, false);
  719. _view_cache_set_attr(view_cache_ctx.dsc->num + 1, 0, true, false);
  720. view_cache_ctx.shrunk = 1;
  721. return 0;
  722. }
  723. static void _view_cache_restore(void)
  724. {
  725. bool is_cross = (view_cache_ctx.focus_idx >= view_cache_ctx.dsc->num);
  726. uint16_t view_id = _view_cache_get_view_id(view_cache_ctx.focus_idx);
  727. SYS_LOG_INF("view_cache: restore view %d", view_id);
  728. if (is_cross) {
  729. /* do not double-set the attr */
  730. if (view_cache_ctx.dsc->cross_attached_view == VIEW_INVALID_ID) {
  731. uint8_t cross_attr[2] = {
  732. _view_cache_decide_attr_cross(view_cache_ctx.dsc->num),
  733. _view_cache_decide_attr_cross(view_cache_ctx.dsc->num + 1),
  734. };
  735. /* restore the drag attr */
  736. _view_cache_set_attr(view_cache_ctx.dsc->num, cross_attr[0], true, false);
  737. _view_cache_set_attr(view_cache_ctx.dsc->num + 1, cross_attr[1], true, false);
  738. }
  739. } else {
  740. if (view_cache_ctx.dsc->cross_attached_view == VIEW_INVALID_ID ||
  741. view_id == view_cache_ctx.dsc->cross_attached_view) {
  742. uint8_t cross_attr[2] = {
  743. [0] = _view_cache_decide_attr_cross(view_cache_ctx.dsc->num),
  744. [1] = _view_cache_decide_attr_cross(view_cache_ctx.dsc->num + 1),
  745. };
  746. _view_cache_set_attr(view_cache_ctx.dsc->num, cross_attr[0], true, false);
  747. }
  748. }
  749. _view_cache_set_focus_idx(view_cache_ctx.main_idx, true);
  750. view_cache_ctx.shrunk = 0;
  751. }
  752. #endif /* CONFIG_VIEW_SCROLL_MEM_DEFAULT */