Bundles / src / view /

view_sink.c

  1. /*
  2. * Copyright (c) 2016 Samsung Electronics Co., Ltd
  3. *
  4. * Licensed under the Flora License, Version 1.1 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://floralicense.org/license/
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16.  
  17. #include <app.h>
  18. #include <efl_extension.h>
  19. #include "$(appName).h"
  20. #include "view/view_common.h"
  21. #include "view_defines.h"
  22. #include "view/view_source.h"
  23. #include "view/view_list.h"
  24. #include "view/view_sink.h"
  25.  
  26. #define FORMAT(type) "%s: "type"<align=right><type>Type: %s</type></align=right>"
  27.  
  28. typedef struct {
  29. char *val_1;
  30. char *val_2;
  31. bool state;
  32. } list_data_t;
  33.  
  34. static struct view_info {
  35. Evas_Object *layout;
  36. Evas_Object *list;
  37. Elm_Genlist_Item_Class *itc_title;
  38. Elm_Genlist_Item_Class *itc_data;
  39. Elm_Genlist_Item_Class *itc_header;
  40. Elm_Genlist_Item_Class *itc_header_data;
  41. Elm_Object_Item *title;
  42.  
  43. } s_info = {
  44. .layout = NULL,
  45. .list = NULL,
  46. .itc_title = NULL,
  47. .itc_data = NULL,
  48. .itc_header = NULL,
  49. .itc_header_data = NULL,
  50. .title = NULL,
  51. };
  52.  
  53. static bool _create_list(void);
  54. static Evas_Object *_item_title_get_cb(void *data, Evas_Object *obj, const char *part);
  55. static Evas_Object *_item_data_get_cb(void *data, Evas_Object *obj, const char *part);
  56. static Evas_Object *_item_header_get_cb(void *data, Evas_Object *obj, const char *part);
  57. static Evas_Object *_item_header_data_get_cb(void *data, Evas_Object *obj, const char *part);
  58. static void _tree_item_expanded(void *data, Evas_Object *obj, void *event_info);
  59. static void _item_delete_cb(void *data, Evas_Object *obj);
  60.  
  61. /**
  62. * @brief Function used to create the sink view.
  63. * @param[in] parent The parent widget
  64. * @return true on success or false on error.
  65. */
  66. bool view_sink_create(Evas_Object *parent)
  67. {
  68. s_info.layout = view_create_layout(parent, EDJ_SINK_FILE_NAME, GROUP_SINK);
  69. if (!s_info.layout)
  70. return false;
  71.  
  72. if (!_create_list())
  73. return false;
  74.  
  75. return true;
  76. }
  77.  
  78. /**
  79. * @brief Returns the sink view's layout widget.
  80. * @return The layout widget.
  81. */
  82. Evas_Object *view_sink_get(void)
  83. {
  84. return s_info.layout;
  85. }
  86.  
  87. /**
  88. * @brief Callback function to be used by the data module. Invoked when a new message is received.
  89. * @param[in] index The index of the new message.
  90. */
  91. void view_sink_new_message_cb(int index)
  92. {
  93. dlog_print(DLOG_INFO, LOG_TAG, "[%s:%d] Message received. Index: %d", __FILE__, __LINE__, index);
  94. s_info.title = view_list_add_item(s_info.list, s_info.itc_title, ELM_GENLIST_ITEM_TREE, NULL, (void*)index, false, ELM_OBJECT_SELECT_MODE_DEFAULT, NULL, NULL);
  95. }
  96.  
  97. /**
  98. * @brief Internal function that creates a list object.
  99. * @return true on success or false on fail.
  100. */
  101. static bool _create_list(void)
  102. {
  103. s_info.list = view_list_create(s_info.layout);
  104. if (!s_info.list) {
  105. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] s_info.list == NULL", __FILE__, __LINE__);
  106. return false;
  107. }
  108.  
  109. s_info.itc_title = view_list_add_itc(_item_title_get_cb, NULL);
  110. if (!s_info.itc_title) {
  111. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] s_info.itc == NULL", __FILE__, __LINE__);
  112. return false;
  113. }
  114.  
  115. s_info.itc_data = view_list_add_itc(_item_data_get_cb, _item_delete_cb);
  116. if (!s_info.itc_data) {
  117. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] s_info.itc_sub == NULL", __FILE__, __LINE__);
  118. return false;
  119. }
  120.  
  121. s_info.itc_header = view_list_add_itc(_item_header_get_cb, NULL);
  122. if (!s_info.itc_header) {
  123. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] s_info.itc_sub == NULL", __FILE__, __LINE__);
  124. return false;
  125. }
  126.  
  127. s_info.itc_header_data = view_list_add_itc(_item_header_data_get_cb, view_list_del_item_cb);
  128. if (!s_info.itc_header_data) {
  129. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] s_info.itc_tree == NULL", __FILE__, __LINE__);
  130. return false;
  131. }
  132.  
  133. view_list_add_tree_callbacks(s_info.list, s_info.itc_title, s_info.itc_data, _tree_item_expanded, view_common_tree_item_contracted_cb);
  134.  
  135. if (!elm_layout_content_set(s_info.layout, PART_LIST, s_info.list)) {
  136. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] Failed to set entry as a swallow %s", __FILE__, __LINE__, PART_LIST);
  137. return false;
  138. }
  139.  
  140. return true;
  141. }
  142.  
  143. /**
  144. * @brief Internal callback function added to an genlist's item class. It is invoked when a title item is realized.
  145. * @param[in] data The user data.
  146. * @param[in] obj A genlist object.
  147. * @param[in] part The part name.
  148. * @return A new layout to be embedded by the genlist's item.
  149. */
  150. static Evas_Object *_item_title_get_cb(void *data, Evas_Object *obj, const char *part)
  151. {
  152. char buf[NAME_MAX] = {0,};
  153. int index = (int)data;
  154. int count = data_get_count(index);
  155.  
  156. Evas_Object *item_layout = view_create_layout(obj, EDJ_SINK_FILE_NAME, GROUP_TITLE);
  157. if (!item_layout) {
  158. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] item_layout == NULL", __FILE__, __LINE__);
  159. return NULL;
  160. }
  161.  
  162. snprintf(buf, NAME_MAX, "Bundle items: %d", count);
  163. elm_layout_text_set(item_layout, PART_ITEM, buf);
  164.  
  165. return item_layout;
  166. }
  167.  
  168. /**
  169. * @brief Internal callback function added to an genlist's item class. It is invoked when a data item (containing the key, value and type) is realized.
  170. * @param[in] data The user data.
  171. * @param[in] obj A genlist object.
  172. * @param[in] part The part name.
  173. * @return A new layout to be embedded by the genlist's item.
  174. */
  175. static Evas_Object *_item_data_get_cb(void *data, Evas_Object *obj, const char *part)
  176. {
  177. list_data_t *list_data = (list_data_t *)data;
  178. char buf[NAME_MAX] = {0,};
  179.  
  180. Evas_Object *item_layout = view_create_layout(obj, EDJ_SINK_FILE_NAME, GROUP_ITEM);
  181. if (!item_layout) {
  182. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] item_layout == NULL", __FILE__, __LINE__);
  183. return NULL;
  184. }
  185.  
  186. snprintf(buf, NAME_MAX, "Type: %s", list_data->val_2);
  187. elm_layout_text_set(item_layout, PART_ITEM_KEY_VAL, list_data->val_1);
  188. elm_layout_text_set(item_layout, PART_ITEM_TYPE, buf);
  189.  
  190. if (list_data->state)
  191. elm_layout_signal_emit(item_layout, SIGNAL_BUBBLES_HIDE, "");
  192.  
  193. return item_layout;
  194. }
  195.  
  196. /**
  197. * @brief Internal callback function added to an genlist's item class. It is invoked when a header item is realized.
  198. * @param[in] data The user data.
  199. * @param[in] obj A genlist object.
  200. * @param[in] part The part name.
  201. * @return A new layout to be embedded by the genlist's item.
  202. */
  203. static Evas_Object *_item_header_get_cb(void *data, Evas_Object *obj, const char *part)
  204. {
  205. Evas_Object *item_layout = view_create_layout(obj, EDJ_SINK_FILE_NAME, GROUP_HEADER);
  206. if (!item_layout) {
  207. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] item_layout == NULL", __FILE__, __LINE__);
  208. return NULL;
  209. }
  210.  
  211. return item_layout;
  212. }
  213.  
  214. /**
  215. * @brief Internal callback function added to an genlist's item class. It is invoked when a header data item is realized.
  216. * @param[in] data The user data.
  217. * @param[in] obj A genlist object.
  218. * @param[in] part The part name.
  219. * @return A new layout to be embedded by the genlist's item.
  220. */
  221. static Evas_Object *_item_header_data_get_cb(void *data, Evas_Object *obj, const char *part)
  222. {
  223. char *text = (char *)data;
  224. char buf[NAME_MAX] = {0,};
  225. char **tokens = NULL;
  226.  
  227. Evas_Object *item_layout = view_create_layout(obj, EDJ_SINK_FILE_NAME, GROUP_HEADER_DATA);
  228. if (!item_layout) {
  229. dlog_print(DLOG_ERROR, LOG_TAG, "[%s:%d] item_layout == NULL", __FILE__, __LINE__);
  230. return NULL;
  231. }
  232.  
  233. tokens = eina_str_split(text, "\2", 2);
  234. elm_layout_text_set(item_layout, PART_ITEM_KEY_VAL, tokens[0]);
  235. snprintf(buf, NAME_MAX, "Type: %s", tokens[1]);
  236. elm_layout_text_set(item_layout, PART_ITEM_TYPE, buf);
  237. return item_layout;
  238. }
  239.  
  240. /**
  241. * @brief Internal function that adds header and data items to the genlist.
  242. * @param[in] index The index of a message to use.
  243. * @param[in] parent The parent item.
  244. */
  245. static void _add_data_items(int index, Elm_Object_Item *parent)
  246. {
  247. Eina_List* l;
  248. char* key;
  249. list_data_t *data = NULL;
  250. Eina_List *keys = data_get_keys(index);
  251. int byte_val;
  252. char buf[NAME_MAX];
  253. char *string_val;
  254. bool state = !data_key_exists(index, BUNDLE_HEADER_KEY); /* If header exists the first item won't display the bubbles */
  255.  
  256. EINA_LIST_FOREACH(keys, l, key) {
  257. if (string_equal(BUNDLE_HEADER_KEY, key)) {
  258. view_list_add_item(s_info.list, s_info.itc_header, ELM_GENLIST_ITEM_TREE, parent, (void*) index, true, ELM_OBJECT_SELECT_MODE_DEFAULT, NULL, NULL);
  259. continue;
  260. } else if (string_equal(BUNDLE_HEADER_DATA_KEY, key)) {
  261. continue;
  262. } else if (data_is_byte(index, key)) {
  263. byte_val = data_get_byte(index, key);
  264. snprintf(buf, NAME_MAX, "%s: %d", key, byte_val);
  265. } else {
  266. string_val = data_get_string(index, key);
  267. snprintf(buf, NAME_MAX, "%s: %s", key, string_val);
  268. }
  269.  
  270. data = calloc(1, sizeof(list_data_t));
  271. data->val_1 = strdup(buf);
  272. data->val_2 = strdup(data_get_type_text(index, key));
  273. data->state = state;
  274.  
  275. view_list_add_item(s_info.list, s_info.itc_data, ELM_GENLIST_ITEM_NONE, parent, (void*)data, false, ELM_OBJECT_SELECT_MODE_NONE, NULL, NULL);
  276. state = false;
  277. }
  278.  
  279. EINA_LIST_FREE(keys, key)
  280. free(key);
  281. }
  282.  
  283. /**
  284. * @brief Internal function that adds header data items.
  285. * @param[in] index The index of a message to use.
  286. * @param[in] parent The parent item.
  287. */
  288. static void _add_header_items(int index, Elm_Object_Item *parent)
  289. {
  290. int len = 0;
  291. int i;
  292. const char **keys = data_get_string_array(index, BUNDLE_HEADER_DATA_KEY, &len);
  293.  
  294. for (i = 0; i < len; i++)
  295. view_list_add_item(s_info.list, s_info.itc_header_data, ELM_GENLIST_ITEM_NONE, parent, (void*)strdup(keys[i]), false, ELM_OBJECT_SELECT_MODE_NONE, NULL, NULL);
  296. }
  297.  
  298. /**
  299. * @brief Internal callback function invoked when a TREE item is expanded
  300. * @param[in] data The use data.
  301. * @param[in] obj The genlist.
  302. * @param[in] event_info The expanded item.
  303. */
  304. static void _tree_item_expanded(void *data, Evas_Object *obj, void *event_info)
  305. {
  306. Elm_Object_Item *parent = (Elm_Object_Item *)event_info;
  307. int index = (int)elm_object_item_data_get(parent);
  308. Evas_Object *header = elm_object_item_part_content_get(parent, "elm.swallow.content");
  309.  
  310. elm_layout_signal_emit(header, SIGNAL_ARROW_STATE_CHANGE, "");
  311.  
  312. if (elm_genlist_item_item_class_get(parent) == s_info.itc_title)
  313. _add_data_items(index, parent);
  314. else if (elm_genlist_item_item_class_get(parent) == s_info.itc_header)
  315. _add_header_items(index, parent);
  316. }
  317.  
  318. /**
  319. * @brief Internal callback function invoked when a data item is removed.
  320. * @param[in] data The user data.
  321. * @param[in] obj The genlist.
  322. */
  323. static void _item_delete_cb(void *data, Evas_Object *obj)
  324. {
  325. list_data_t *it_data = (list_data_t *)data;
  326.  
  327. if (!it_data)
  328. return;
  329.  
  330. free(it_data->val_1);
  331. free(it_data->val_2);
  332. free(it_data);
  333. }