2 * Copyright (c) 2017 Samsung Electronics Co., Ltd All Rights Reserved
4 * Licensed under the Apache License, Version 2.0 (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
8 * http://www.apache.org/licenses/LICENSE-2.0
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.
21 #include <bundle_internal.h>
23 #include <aul_widget.h>
26 #include <glib-object.h>
29 #include <widget_errno.h>
30 #include <widget_instance.h>
31 #include <aul_app_com.h>
32 #include <Ecore_Wl2.h>
33 #include <system_info.h>
34 #include <screen_connector_provider.h>
35 #include <appcore_multiwindow_base.h>
37 #include "widget_base.h"
43 #define LOG_TAG "CAPI_WIDGET_APPLICATION"
44 #define APP_TYPE_WIDGET "widgetapp"
45 #define STATUS_FOREGROUND "fg"
46 #define STATUS_BACKGROUND "bg"
48 static int __app_event_converter[APPCORE_BASE_EVENT_MAX] = {
49 [APP_EVENT_LOW_MEMORY] = APPCORE_BASE_EVENT_LOW_MEMORY,
50 [APP_EVENT_LOW_BATTERY] = APPCORE_BASE_EVENT_LOW_BATTERY,
51 [APP_EVENT_LANGUAGE_CHANGED] = APPCORE_BASE_EVENT_LANG_CHANGE,
52 [APP_EVENT_DEVICE_ORIENTATION_CHANGED]
53 = APPCORE_BASE_EVENT_DEVICE_ORIENTATION_CHANGED,
54 [APP_EVENT_REGION_FORMAT_CHANGED] = APPCORE_BASE_EVENT_REGION_CHANGE,
55 [APP_EVENT_SUSPENDED_STATE_CHANGED]
56 = APPCORE_BASE_EVENT_SUSPENDED_STATE_CHANGE,
59 struct app_event_info {
60 app_event_type_e type;
64 struct app_event_handler {
65 app_event_type_e type;
71 struct widget_foreach_context {
72 widget_base_instance_cb callback;
76 typedef struct _widget_base_context {
82 } widget_base_context;
84 typedef struct _widget_base_instance_data {
92 char *pending_content;
94 } widget_base_instance_data;
96 static widget_base_context __context;
98 static char *__package_id;
99 static bool __fg_signal;
100 static char *__viewer_endpoint;
101 static bool __is_permanent;
102 static void __call_update_cb(const char *class_id, const char *id, int force,
103 const char *content_raw);
105 static gboolean __timeout_cb(gpointer user_data)
107 widget_base_instance_data *data =
108 (widget_base_instance_data *)user_data;
109 appcore_multiwindow_base_instance_h cxt;
110 const char *class_id;
112 cxt = appcore_multiwindow_base_instance_find(data->id);
115 LOGE("Can't find the instance");
116 return G_SOURCE_REMOVE;
119 if (appcore_multiwindow_base_instance_is_resumed(cxt)) {
120 LOGD("Periodic update!");
121 class_id = appcore_multiwindow_base_instance_get_class_id(cxt);
122 __call_update_cb(class_id, data->id, 0, NULL);
124 data->pending_update = true;
125 if (data->periodic_timer) {
126 LOGD("Remove timer!");
127 g_source_remove(data->periodic_timer);
128 data->periodic_timer = 0;
132 return G_SOURCE_CONTINUE;
135 static bool __is_widget_feature_enabled(void)
137 static bool feature = false;
138 static bool retrieved = false;
141 if (retrieved == true)
144 ret = system_info_get_platform_bool(FEATURE_SHELL_APPWIDGET, &feature);
145 if (ret != SYSTEM_INFO_ERROR_NONE) {
146 LOGE("failed to get system info"); /* LCOV_EXCL_LINE */
147 return false; /* LCOV_EXCL_LINE */
155 static void __check_empty_instance(void)
157 int cnt = appcore_multiwindow_base_instance_get_cnt();
163 static void __instance_drop(appcore_multiwindow_base_instance_h instance_h)
165 widget_base_instance_data *data;
167 data = appcore_multiwindow_base_instance_get_extra(instance_h);
168 appcore_multiwindow_base_instance_drop(instance_h);
169 free(data->pending_content);
173 __check_empty_instance();
176 static gint __comp_class(gconstpointer a, gconstpointer b)
178 const widget_base_class *cls = a;
180 return strcmp(cls->id, b);
183 static widget_base_class *__get_class(const char *class_id)
185 widget_base_class *cls;
188 class_node = g_list_find_custom(__context.classes, class_id,
190 if (class_node == NULL) {
191 LOGE("empty classes");
194 cls = (widget_base_class *)class_node->data;
199 static int __send_update_status(const char *class_id, const char *instance_id,
200 int status, int err, bundle *extra)
204 aul_widget_send_status_to_viewer(class_id, instance_id,
205 __viewer_endpoint, status, err, extra);
206 lifecycle = widget_instance_convert_event_to_lifecycle_status(status);
207 if (lifecycle > -1) {
208 aul_widget_send_status_to_service(
209 class_id, instance_id, __package_id, lifecycle);
215 static void __control_create(const char *class_id, const char *id, bundle *b)
217 widget_base_instance_data *data;
218 char *content = NULL;
220 if (appcore_multiwindow_base_instance_find(id)) {
221 LOGE("Already exist id (%s)", id);
225 data = (widget_base_instance_data *)
226 calloc(1, sizeof(widget_base_instance_data));
228 LOGE("Out of memory");
232 data->id = strdup(id);
235 /* call stub create */
236 appcore_multiwindow_base_instance_run(class_id, id, data);
237 if (appcore_multiwindow_base_instance_find(id)) {
239 bundle_get_str(b, WIDGET_K_CONTENT_INFO, &content);
241 data->content = strdup(content);
245 static void __control_resume(const char *class_id, const char *id, bundle *b)
247 appcore_multiwindow_base_instance_h cxt;
249 cxt = appcore_multiwindow_base_instance_find(id);
251 LOGE("context not found: %s", id);
255 /* call stub resume */
256 appcore_multiwindow_base_instance_resume(cxt);
259 static void __control_pause(const char *class_id, const char *id, bundle *b)
261 appcore_multiwindow_base_instance_h instance_h;
263 instance_h = appcore_multiwindow_base_instance_find(id);
266 LOGE("instance not found: %s", id);
270 /* call stub pause */
271 appcore_multiwindow_base_instance_pause(instance_h);
274 static void __control_resize(const char *class_id, const char *id, bundle *b)
276 appcore_multiwindow_base_instance_h instance_h;
283 widget_base_class *cls;
284 const appcore_multiwindow_base_class *raw_cls;
286 instance_h = appcore_multiwindow_base_instance_find(id);
288 LOGE("context not found: %s", id);
292 raw_cls = appcore_multiwindow_base_instance_get_class(instance_h);
296 cls = __get_class(class_id);
298 LOGE("class not found: %s", class_id);
301 class_data = raw_cls->data;
302 bundle_get_str(b, WIDGET_K_WIDTH, &w_str);
303 bundle_get_str(b, WIDGET_K_HEIGHT, &h_str);
306 w = (int)g_ascii_strtoll(w_str, &remain, 10);
309 h = (int)g_ascii_strtoll(h_str, &remain, 10);
312 cls->ops.resize(instance_h, w, h, class_data);
314 LOGD("%s is resized to %dx%d", id, w, h);
315 __send_update_status(class_id, id,
316 WIDGET_INSTANCE_EVENT_SIZE_CHANGED, 0, NULL);
319 static void __call_update_cb(const char *class_id, const char *id, int force,
320 const char *content_raw)
323 widget_base_class *cls;
324 const appcore_multiwindow_base_class *raw_cls;
325 appcore_multiwindow_base_instance_h instance_h;
326 bundle *content = NULL;
328 instance_h = appcore_multiwindow_base_instance_find(id);
330 LOGE("context not found: %s", id);
334 raw_cls = appcore_multiwindow_base_instance_get_class(instance_h);
336 LOGE("class is NULL");
340 class_data = raw_cls->data;
341 cls = __get_class(class_id);
343 LOGE("class not found: %s", class_id);
347 if (!cls->ops.update) {
348 LOGE("update callback is NULL");
353 content = bundle_decode((const bundle_raw *)content_raw,
354 strlen(content_raw));
358 cls->ops.update(instance_h, content, force, class_data);
360 __send_update_status(class_id, id,
361 WIDGET_INSTANCE_EVENT_UPDATE, 0, NULL);
362 LOGD("updated:%s", id);
365 bundle_free(content);
368 static void __update_pending_content(
369 appcore_multiwindow_base_instance_h instance_h,
370 const char *content_raw)
372 widget_base_instance_data *data;
374 data = (widget_base_instance_data *)
375 appcore_multiwindow_base_instance_get_extra(instance_h);
377 if (data->pending_content) {
378 free(data->pending_content);
379 data->pending_content = NULL;
383 data->pending_content = strdup(content_raw);
384 if (data->pending_content == NULL)
385 LOGW("Out of memory");
388 data->pending_update = true;
391 static void __update_process(const char *class_id, const char *id,
392 appcore_multiwindow_base_instance_h instance_h, void *data)
394 char *content_raw = NULL;
395 char *force_str = NULL;
400 LOGE("bundle is NULL");
404 bundle_get_str(b, WIDGET_K_FORCE, &force_str);
406 if (force_str && strcmp(force_str, "true") == 0)
411 bundle_get_str(b, WIDGET_K_CONTENT_INFO, &content_raw);
412 if (!appcore_multiwindow_base_instance_is_resumed(instance_h) && !force)
413 __update_pending_content(instance_h, content_raw);
415 __call_update_cb(class_id, id, force, content_raw);
418 static void __control_update(const char *class_id, const char *id, bundle *b)
420 appcore_multiwindow_base_instance_h instance_h;
423 appcore_multiwindow_base_instance_foreach(class_id,
424 __update_process, b);
428 instance_h = appcore_multiwindow_base_instance_find(id);
430 LOGE("context not found: %s", id);
434 __update_process(class_id, id, instance_h, b);
437 static void __control_destroy(const char *class_id, const char *id, bundle *b)
439 appcore_multiwindow_base_instance_h instance_h;
440 widget_base_instance_data *data;
442 instance_h = appcore_multiwindow_base_instance_find(id);
444 LOGE("could not find widget obj: %s, clear amd info", id);
445 aul_widget_instance_del(class_id, id);
449 data = (widget_base_instance_data *)
450 appcore_multiwindow_base_instance_get_extra(instance_h);
453 /* call stub terminate */
454 appcore_multiwindow_base_instance_exit(instance_h);
455 free(data->pending_content);
459 __check_empty_instance();
460 aul_widget_write_log(LOG_TAG,
461 "[%s:%d] instance_id(%s)", __FUNCTION__, __LINE__, id);
464 static void __control_change_period(const char *class_id, const char *id,
467 appcore_multiwindow_base_instance_h instance_h;
468 widget_base_instance_data *data;
469 double *period = NULL;
473 instance_h = appcore_multiwindow_base_instance_find(id);
475 LOGE("context not found: %s", id);
479 data = (widget_base_instance_data *)
480 appcore_multiwindow_base_instance_get_extra(instance_h);
483 LOGE("could not find instance data: %s", id);
487 if (data->periodic_timer) {
488 LOGD("Remove timer!");
489 g_source_remove(data->periodic_timer);
490 data->periodic_timer = 0;
493 ret = bundle_get_byte(b, WIDGET_K_PERIOD, (void **)&period, &size);
494 if (ret == BUNDLE_ERROR_NONE)
495 data->period = *period;
497 if (data->period > 0) {
498 LOGD("Restart timer!");
499 data->periodic_timer = g_timeout_add_seconds(data->period,
506 static int __multiwindow_create(void *data)
508 char pkgid[256] = {0, };
511 appcore_multiwindow_base_on_create();
512 app_get_id(&__appid);
513 if (aul_app_get_pkgid_bypid(getpid(), pkgid, sizeof(pkgid)) == 0)
514 __package_id = strdup(pkgid);
516 if (!__package_id || !__appid) {
517 LOGE("__package_id is NULL");
521 screen_connector_provider_init();
522 if (__context.ops.create)
523 ret = __context.ops.create(data);
525 LOGD("widget base is created");
529 static int __multiwindow_terminate(void *data)
531 if (__context.ops.terminate)
532 __context.ops.terminate(data);
533 screen_connector_provider_fini();
535 if (__viewer_endpoint) {
536 free(__viewer_endpoint);
537 __viewer_endpoint = NULL;
550 appcore_multiwindow_base_on_terminate();
552 LOGD("widget base is terminated");
556 static int __multiwindow_control(bundle *b, void *data)
558 char *class_id = NULL;
560 char *operation = NULL;
562 appcore_multiwindow_base_on_control(b);
563 bundle_get_str(b, WIDGET_K_CLASS, &class_id);
564 /* for previous version compatibility, use appid for default class id */
565 if (class_id == NULL)
568 bundle_get_str(b, AUL_K_WIDGET_INSTANCE_ID, &id);
569 bundle_get_str(b, WIDGET_K_OPERATION, &operation);
572 LOGE("operation is NULL");
576 LOGI("app control operation(%s)", operation);
577 if (strcmp(operation, "create") == 0)
578 __control_create(class_id, id, b);
579 else if (strcmp(operation, "resize") == 0)
580 __control_resize(class_id, id, b);
581 else if (strcmp(operation, "update") == 0)
582 __control_update(class_id, id, b);
583 else if (strcmp(operation, "destroy") == 0)
584 __control_destroy(class_id, id, b);
585 else if (strcmp(operation, "resume") == 0)
586 __control_resume(class_id, id, b);
587 else if (strcmp(operation, "pause") == 0)
588 __control_pause(class_id, id, b);
589 else if (strcmp(operation, "terminate") == 0)
590 __control_destroy(class_id, id, b);
591 else if (strcmp(operation, "period") == 0)
592 __control_change_period(class_id, id, b);
597 static void __inst_resume_cb(const char *class_id, const char *id,
598 appcore_multiwindow_base_instance_h cxt, void *data)
600 __control_resume(class_id, id, data);
603 static void __get_content(bundle *b)
605 char *instance_id = NULL;
606 appcore_multiwindow_base_instance_h cxt;
607 widget_base_instance_data * we;
609 bundle_get_str(b, AUL_K_WIDGET_INSTANCE_ID, &instance_id);
611 LOGE("instance id is NULL");
615 cxt = appcore_multiwindow_base_instance_find(instance_id);
617 LOGE("could not find widget obj: %s", instance_id);
621 we = appcore_multiwindow_base_instance_get_extra(cxt);
623 LOGE("widget extra is NULL");
628 bundle_add_str(b, AUL_K_WIDGET_CONTENT_INFO, we->content);
629 LOGD("content info of %s found", instance_id);
631 bundle_add_str(b, AUL_K_WIDGET_CONTENT_INFO, "");
632 LOGD("empty content info added");
636 static int __multiwindow_receive(aul_type type, bundle *b, void *data)
638 appcore_multiwindow_base_on_receive(type, b);
642 appcore_multiwindow_base_instance_foreach_full(
643 __inst_resume_cb, b);
648 case AUL_WIDGET_CONTENT:
658 static void __multiwindow_init(int argc, char **argv, void *data)
660 if (__context.ops.init)
661 __context.ops.init(argc, argv, data);
664 static void __multiwindow_finish(void)
666 if (__context.ops.finish) {
667 __context.ops.finish();
668 /* Check Loader case */
669 if (getenv("AUL_LOADER_INIT")) {
670 unsetenv("AUL_LOADER_INIT");
671 __context.ops.finish();
676 static void __multiwindow_run(void *data)
678 if (__context.ops.run)
679 __context.ops.run(data);
682 static void __multiwindow_exit(void *data)
684 if (__context.ops.exit)
685 __context.ops.exit(data);
688 static void __multiwindow_trim_memory(void *data)
690 if (__context.ops.trim_memory)
691 __context.ops.trim_memory(data);
694 EXPORT_API int widget_base_exit(void)
699 appcore_multiwindow_base_exit();
700 cnt = appcore_multiwindow_base_instance_get_cnt();
701 if (cnt == 0 && __is_permanent)
702 ret = aul_notify_exit();
704 aul_widget_write_log(LOG_TAG,
705 "[%s:%d] exit : ret(%d), cnt(%d), permanent(%d)",
706 __FUNCTION__, __LINE__, ret, cnt, __is_permanent);
711 static gboolean __finish_event_cb(gpointer user_data)
713 appcore_multiwindow_base_instance_h cxt = user_data;
716 const char *class_id;
719 LOGE("user_data is NULL");
723 id = appcore_multiwindow_base_instance_get_id(cxt);
724 class_id = appcore_multiwindow_base_instance_get_class_id(cxt);
728 LOGE("Out-of-memory");
732 bundle_add_str(b, WIDGET_K_OPERATION, "terminate");
733 __control_destroy(class_id, id, b);
739 EXPORT_API int widget_base_terminate_context(widget_base_instance_h context)
741 if (!__is_widget_feature_enabled()) {
742 LOGE("not supported"); /* LCOV_EXCL_LINE */
743 return WIDGET_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
747 LOGE("context is null");
748 return WIDGET_ERROR_INVALID_PARAMETER;
751 g_idle_add(__finish_event_cb, context);
753 return WIDGET_ERROR_NONE;
756 static void __inst_full_cb(const char *class_id, const char *id,
757 appcore_multiwindow_base_instance_h cxt, void *data)
759 struct widget_foreach_context *foreach_context = data;
764 if (foreach_context->callback)
765 foreach_context->callback(cxt, foreach_context->data);
768 EXPORT_API int widget_base_foreach_context(widget_base_instance_cb cb, void *data)
770 struct widget_foreach_context foreach_context;
772 if (!__is_widget_feature_enabled()) {
773 LOGE("not supported"); /* LCOV_EXCL_LINE */
774 return WIDGET_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
778 LOGE("callback is NULL");
779 return WIDGET_ERROR_INVALID_PARAMETER;
782 foreach_context.callback = cb;
783 foreach_context.data = data;
784 appcore_multiwindow_base_instance_foreach_full(__inst_full_cb, &foreach_context);
786 return WIDGET_ERROR_NONE;
789 static int __event_cb(void *event, void *data)
791 app_event_handler_h handler = data;
793 struct app_event_info app_event;
795 app_event.type = handler->type;
796 app_event.value = event;
799 handler->cb(&app_event, handler->data);
804 EXPORT_API int widget_base_add_event_handler(app_event_handler_h *event_handler,
805 app_event_type_e event_type,
806 app_event_cb callback,
811 app_event_handler_h handler;
813 r = system_info_get_platform_bool(FEATURE_SHELL_APPWIDGET, &feature);
815 return WIDGET_BASE_ERROR_FAULT;
818 return WIDGET_BASE_ERROR_NOT_SUPPORTED;
820 if (event_handler == NULL || callback == NULL)
821 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
823 if (event_type < APP_EVENT_LOW_MEMORY
824 || event_type > APP_EVENT_REGION_FORMAT_CHANGED)
825 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
827 if (event_type == APP_EVENT_DEVICE_ORIENTATION_CHANGED)
828 return WIDGET_BASE_ERROR_NOT_SUPPORTED;
831 handler = calloc(1, sizeof(struct app_event_handler));
833 return WIDGET_BASE_ERROR_OUT_OF_MEMORY;
835 handler->type = event_type;
836 handler->cb = callback;
837 handler->data = user_data;
838 handler->raw = appcore_base_add_event(
839 __app_event_converter[event_type], __event_cb, handler);
840 *event_handler = handler;
842 return WIDGET_BASE_ERROR_NONE;
845 EXPORT_API int widget_base_remove_event_handler(app_event_handler_h
850 app_event_type_e type;
852 r = system_info_get_platform_bool(FEATURE_SHELL_APPWIDGET, &feature);
854 return WIDGET_BASE_ERROR_FAULT;
857 return WIDGET_BASE_ERROR_NOT_SUPPORTED;
859 if (event_handler == NULL)
860 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
862 type = event_handler->type;
863 if (type < APP_EVENT_LOW_MEMORY ||
864 type > APP_EVENT_REGION_FORMAT_CHANGED)
865 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
867 r = appcore_base_remove_event(event_handler->raw);
869 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
873 return WIDGET_BASE_ERROR_NONE;
876 EXPORT_API int widget_base_context_set_content_info(
877 widget_base_instance_h context,
878 bundle *content_info)
881 bundle_raw *raw = NULL;
884 const char *class_id;
885 widget_base_instance_data *data;
886 appcore_multiwindow_base_instance_h instance_h;
888 if (!__is_widget_feature_enabled()) {
889 LOGE("not supported"); /* LCOV_EXCL_LINE */
890 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
893 if (!context || !content_info)
894 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
896 instance_h = (appcore_multiwindow_base_instance_h)context;
897 id = appcore_multiwindow_base_instance_get_id(instance_h);
898 class_id = appcore_multiwindow_base_instance_get_class_id(instance_h);
899 data = appcore_multiwindow_base_instance_get_extra(instance_h);
901 if (!class_id || !id || !data)
902 return WIDGET_BASE_ERROR_FAULT;
904 ret = __send_update_status(class_id, id,
905 WIDGET_INSTANCE_EVENT_EXTRA_UPDATED, 0, content_info);
910 bundle_encode(content_info, &raw, &len);
912 data->content = strdup((const char *)raw);
914 data->content = NULL;
918 /* LCOV_EXCL_START */
919 LOGE("failed to send content info: %s of %s (%d)", id,
921 return WIDGET_BASE_ERROR_IO_ERROR;
925 return WIDGET_BASE_ERROR_NONE;
928 EXPORT_API int widget_base_context_get_tag(widget_base_instance_h context, void **tag)
930 appcore_multiwindow_base_instance_h instance_h;
931 widget_base_instance_data *data;
933 if (!__is_widget_feature_enabled()) {
934 LOGE("not supported"); /* LCOV_EXCL_LINE */
935 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
938 if (!context || !tag) {
939 LOGE("Invalid parameter");
940 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
943 instance_h = (appcore_multiwindow_base_instance_h)context;
944 data = (widget_base_instance_data *)
945 appcore_multiwindow_base_instance_get_extra(instance_h);
948 LOGE("Invalid parameter");
949 return WIDGET_ERROR_INVALID_PARAMETER;
954 return WIDGET_BASE_ERROR_NONE;
957 EXPORT_API int widget_base_context_set_tag(widget_base_instance_h context, void *tag)
959 appcore_multiwindow_base_instance_h instance_h;
960 widget_base_instance_data *data;
962 if (!__is_widget_feature_enabled()) {
963 LOGE("not supported"); /* LCOV_EXCL_LINE */
964 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
968 LOGE("Invalid parameter");
969 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
972 instance_h = (appcore_multiwindow_base_instance_h)context;
973 data = (widget_base_instance_data *)
974 appcore_multiwindow_base_instance_get_extra(instance_h);
977 return WIDGET_BASE_ERROR_NONE;
980 EXPORT_API void *widget_base_context_get_user_data(
981 widget_base_instance_h context)
983 appcore_multiwindow_base_instance_h instance_h;
984 widget_base_instance_data *data;
986 if (!__is_widget_feature_enabled()) {
987 LOGE("not supported"); /* LCOV_EXCL_LINE */
988 return NULL; /* LCOV_EXCL_LINE */
992 LOGE("Invalid parameter");
996 instance_h = (appcore_multiwindow_base_instance_h)context;
997 data = (widget_base_instance_data *)
998 appcore_multiwindow_base_instance_get_extra(instance_h);
1000 return data->user_data;
1004 EXPORT_API int widget_base_context_set_user_data(
1005 widget_base_instance_h context, void *user_data)
1007 appcore_multiwindow_base_instance_h instance_h;
1008 widget_base_instance_data *data;
1010 if (!__is_widget_feature_enabled()) {
1011 LOGE("not supported"); /* LCOV_EXCL_LINE */
1012 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
1016 LOGE("Invalid parameter");
1017 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
1020 instance_h = (appcore_multiwindow_base_instance_h)context;
1021 data = (widget_base_instance_data *)
1022 appcore_multiwindow_base_instance_get_extra(instance_h);
1023 data->user_data = user_data;
1025 return WIDGET_BASE_ERROR_NONE;
1028 EXPORT_API int widget_base_context_get_id(widget_base_instance_h context, char **id)
1030 appcore_multiwindow_base_instance_h instance_h;
1032 if (!__is_widget_feature_enabled()) {
1033 LOGE("not supported"); /* LCOV_EXCL_LINE */
1034 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
1037 instance_h = (appcore_multiwindow_base_instance_h)context;
1038 *id = (char *)appcore_multiwindow_base_instance_get_id(instance_h);
1040 return WIDGET_BASE_ERROR_NONE;
1043 EXPORT_API const char *widget_base_get_viewer_endpoint()
1045 return __viewer_endpoint;
1048 EXPORT_API int widget_base_init(widget_base_ops ops, int argc, char **argv,
1052 char *viewer_endpoint = NULL;
1053 appcore_multiwindow_base_ops raw_ops
1054 = appcore_multiwindow_base_get_default_ops();
1056 __context.ops = ops;
1057 __context.argc = argc;
1058 __context.argv = argv;
1059 __context.data = data;
1061 /* override methods */
1062 raw_ops.base.create = __multiwindow_create;
1063 raw_ops.base.control = __multiwindow_control;
1064 raw_ops.base.terminate = __multiwindow_terminate;
1065 raw_ops.base.receive = __multiwindow_receive;
1066 raw_ops.base.init = __multiwindow_init;
1067 raw_ops.base.finish = __multiwindow_finish;
1068 raw_ops.base.run = __multiwindow_run;
1069 raw_ops.base.exit = __multiwindow_exit;
1070 raw_ops.base.trim_memory = __multiwindow_trim_memory;
1072 if (!__is_widget_feature_enabled()) {
1073 LOGE("not supported"); /* LCOV_EXCL_LINE */
1074 return WIDGET_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
1077 kb = bundle_import_from_argv(argc, argv);
1079 bundle_get_str(kb, AUL_K_WIDGET_VIEWER, &viewer_endpoint);
1080 if (viewer_endpoint) {
1081 LOGD("viewer endpoint :%s", viewer_endpoint);
1082 __viewer_endpoint = strdup(viewer_endpoint);
1084 LOGE("endpoint is missing");
1089 LOGE("failed to get launch argv"); /* LCOV_EXCL_LINE */
1090 return WIDGET_ERROR_FAULT;
1093 if (appcore_multiwindow_base_init(raw_ops, argc, argv, data) < 0)
1094 return WIDGET_ERROR_FAULT;
1096 return WIDGET_ERROR_NONE;
1099 static int __on_create(void *data)
1101 return widget_base_on_create();
1104 static int __on_terminate(void *data)
1106 return widget_base_on_terminate();
1109 static void __on_init(int argc, char **argv, void *data)
1111 widget_base_on_init(argc, argv);
1114 static void __on_finish(void)
1116 widget_base_on_finish();
1119 static void __on_run(void *data)
1121 widget_base_on_run();
1124 static void __on_exit(void *data)
1126 widget_base_on_exit();
1129 static void __on_trim_memory(void *data)
1131 widget_base_on_trim_memory();
1134 EXPORT_API int widget_base_on_create(void)
1136 appcore_multiwindow_base_on_create();
1141 EXPORT_API int widget_base_on_terminate(void)
1143 appcore_multiwindow_base_on_terminate();
1148 EXPORT_API int widget_base_on_init(int argc, char **argv)
1153 EXPORT_API void widget_base_on_finish(void)
1157 EXPORT_API void widget_base_on_run(void)
1161 EXPORT_API void widget_base_on_exit(void)
1165 EXPORT_API int widget_base_on_trim_memory(void)
1167 appcore_multiwindow_base_on_trim_memory();
1172 EXPORT_API widget_base_ops widget_base_get_default_ops(void)
1174 widget_base_ops ops;
1176 /* override methods */
1177 ops.create = __on_create;
1178 ops.terminate = __on_terminate;
1179 ops.init = __on_init;
1180 ops.finish = __on_finish;
1182 ops.exit = __on_exit;
1183 ops.trim_memory = __on_trim_memory;
1188 static void __free_class(gpointer data)
1190 widget_base_class *cls = data;
1196 EXPORT_API void widget_base_fini(void)
1198 appcore_multiwindow_base_fini();
1199 g_list_free_full(__context.classes, __free_class);
1200 __context.classes = NULL;
1203 EXPORT_API int widget_base_context_window_bind(
1204 widget_base_instance_h instance_h, const char *id,
1205 Ecore_Wl2_Window *wl_win)
1207 struct wl_surface *surface;
1209 surface = ecore_wl2_window_surface_get(wl_win);
1210 if (surface == NULL) {
1211 LOGE("failed to get surface"); /* LCOV_EXCL_LINE */
1212 return WIDGET_BASE_ERROR_FAULT; /* LCOV_EXCL_LINE */
1215 screen_connector_provider_remote_enable(id, surface);
1216 appcore_multiwindow_base_window_bind(instance_h, wl_win);
1218 return WIDGET_BASE_ERROR_NONE;
1221 static int __class_on_create(widget_base_instance_h instance_h, bundle *content,
1222 int w, int h, void *class_data)
1224 return widget_base_class_on_create(instance_h, content, w, h);
1227 static int __class_on_resume(widget_base_instance_h instance_h, void *class_data)
1229 return widget_base_class_on_resume(instance_h);
1232 static int __class_on_pause(widget_base_instance_h instance_h,
1235 return widget_base_class_on_pause(instance_h);
1238 static int __class_on_resize(widget_base_instance_h instance_h, int w, int h,
1241 return widget_base_class_on_resize(instance_h, w, h);
1244 static int __class_on_update(widget_base_instance_h instance_h, bundle *content,
1245 int force, void *class_data)
1247 return widget_base_class_on_update(instance_h, content, force);
1250 static int __class_on_destroy(widget_base_instance_h instance_h,
1251 widget_base_destroy_type_e reason, bundle *content,
1254 return widget_base_class_on_destroy(instance_h, reason, content);
1257 static void __multiwindow_instance_create(
1258 appcore_multiwindow_base_instance_h instance_h,
1261 widget_base_instance_data *instance_data;
1263 bundle *content_info = NULL;
1265 char *class_id = NULL;
1266 char *operation = NULL;
1267 char *content = NULL;
1270 char *remain = NULL;
1274 widget_base_class *cls;
1275 double *period = NULL;
1278 appcore_multiwindow_base_class_on_create(instance_h);
1279 instance_data = appcore_multiwindow_base_instance_get_extra(instance_h);
1280 b = instance_data->args;
1282 bundle_get_str(b, WIDGET_K_CLASS, &class_id);
1283 /* for previous version compatibility, use appid for default class id */
1284 if (class_id == NULL)
1287 cls = __get_class(class_id);
1289 LOGE("class not found: %s", class_id);
1293 bundle_get_str(b, AUL_K_WIDGET_INSTANCE_ID, &id);
1294 bundle_get_str(b, WIDGET_K_OPERATION, &operation);
1297 LOGE("no operation provided");
1301 bundle_get_str(b, WIDGET_K_CONTENT_INFO, &content);
1302 bundle_get_str(b, WIDGET_K_WIDTH, &w_str);
1303 bundle_get_str(b, WIDGET_K_HEIGHT, &h_str);
1306 w = (int)g_ascii_strtoll(w_str, &remain, 10);
1309 h = (int)g_ascii_strtoll(h_str, &remain, 10);
1312 content_info = bundle_decode((const bundle_raw *)content,
1315 if (cls->ops.create)
1316 ret = cls->ops.create(instance_h, content_info, w, h, class_data);
1319 LOGW("Create callback returns error(%d)", ret);
1320 ret = __send_update_status(class_id, id,
1321 WIDGET_INSTANCE_EVENT_CREATE_ABORTED, ret, NULL);
1323 LOGE("Fail to send abort status (%d) ", ret);
1324 __instance_drop(instance_h);
1326 LOGD("%s is created", id);
1327 aul_widget_instance_add(class_id, id);
1328 ret = __send_update_status(class_id, id,
1329 WIDGET_INSTANCE_EVENT_CREATE, 0, NULL);
1331 LOGE("Fail to send create status (%d) ", ret);
1333 ret = bundle_get_byte(b, WIDGET_K_PERIOD, (void **)&period,
1335 if (ret == BUNDLE_ERROR_NONE && *period > 0) {
1336 LOGI("set periodic update timer (%lf)", *period);
1337 instance_data->period = *period;
1338 instance_data->periodic_timer = g_timeout_add_seconds(
1339 instance_data->period,
1340 __timeout_cb, instance_data);
1345 bundle_free(content_info);
1348 static void __multiwindow_instance_resume(
1349 appcore_multiwindow_base_instance_h instance_h,
1353 const char *class_id;
1354 widget_base_class *cls;
1355 widget_base_instance_data *data;
1357 appcore_multiwindow_base_class_on_resume(instance_h);
1358 id = appcore_multiwindow_base_instance_get_id(instance_h);
1359 class_id = appcore_multiwindow_base_instance_get_class_id(instance_h);
1360 cls = __get_class(class_id);
1362 LOGE("class not found: %s", class_id);
1366 data = (widget_base_instance_data *)
1367 appcore_multiwindow_base_instance_get_extra(instance_h);
1369 if (data->pending_update) {
1370 LOGD("pending update!");
1371 data->pending_update = false;
1372 __call_update_cb(class_id, data->id, 0, data->pending_content);
1373 if (data->period > 0) {
1374 LOGD("Restart timer!");
1375 data->periodic_timer = g_timeout_add_seconds(
1377 __timeout_cb, data);
1381 if (cls->ops.resume)
1382 cls->ops.resume(instance_h, class_data);
1384 LOGD("%s is resumed", id);
1385 __send_update_status(class_id, id,
1386 WIDGET_INSTANCE_EVENT_RESUME, 0, NULL);
1389 LOGD("Send fg signal to resourceD");
1390 aul_widget_instance_change_status(class_id, STATUS_FOREGROUND);
1395 static void __multiwindow_instance_pause(
1396 appcore_multiwindow_base_instance_h instance_h,
1400 const char *class_id;
1401 widget_base_class *cls;
1403 appcore_multiwindow_base_class_on_pause(instance_h);
1404 id = appcore_multiwindow_base_instance_get_id(instance_h);
1405 class_id = appcore_multiwindow_base_instance_get_class_id(instance_h);
1406 cls = __get_class(class_id);
1408 LOGE("class not found: %s", class_id);
1413 cls->ops.pause(instance_h, class_data);
1415 LOGD("%s is paused", id);
1416 __send_update_status(class_id, id,
1417 WIDGET_INSTANCE_EVENT_PAUSE, 0, NULL);
1420 LOGD("Send bg signal to resourceD");
1421 aul_widget_instance_change_status(class_id, STATUS_BACKGROUND);
1422 __fg_signal = false;
1426 static void __multiwindow_instance_terminate(
1427 appcore_multiwindow_base_instance_h instance_h,
1430 widget_base_instance_data *data;
1432 char *operation = NULL;
1433 bundle *content_info;
1434 widget_base_destroy_type_e reason = WIDGET_BASE_DESTROY_TYPE_TEMPORARY;
1435 int event = WIDGET_INSTANCE_EVENT_TERMINATE;
1437 const char *class_id;
1438 widget_base_class *cls;
1440 id = appcore_multiwindow_base_instance_get_id(instance_h);
1441 class_id = appcore_multiwindow_base_instance_get_class_id(instance_h);
1442 data = appcore_multiwindow_base_instance_get_extra(
1443 (appcore_multiwindow_base_instance_h)instance_h);
1445 cls = __get_class(class_id);
1447 LOGE("class not found: %s", class_id);
1452 bundle_get_str(b, WIDGET_K_OPERATION, &operation);
1453 if (operation && strcmp(operation, "destroy") == 0)
1454 reason = WIDGET_BASE_DESTROY_TYPE_PERMANENT;
1458 content_info = bundle_decode((const bundle_raw *)data->content,
1459 strlen(data->content));
1461 content_info = bundle_create();
1463 if (cls->ops.destroy)
1464 cls->ops.destroy(instance_h, reason, content_info, class_data);
1466 LOGW("%s is destroyed %d", id, reason);
1467 if (reason == WIDGET_BASE_DESTROY_TYPE_PERMANENT) {
1468 __is_permanent = true;
1469 event = WIDGET_INSTANCE_EVENT_DESTROY;
1470 aul_widget_instance_del(class_id, id);
1472 __is_permanent = false;
1473 __send_update_status(class_id, id,
1474 WIDGET_INSTANCE_EVENT_EXTRA_UPDATED, 0,
1479 bundle_free(content_info);
1481 if (data->periodic_timer)
1482 g_source_remove(data->periodic_timer);
1484 __send_update_status(class_id, id, event, 0, NULL);
1485 appcore_multiwindow_base_class_on_terminate(instance_h);
1488 EXPORT_API int widget_base_class_on_create(widget_base_instance_h instance_h,
1489 bundle *content, int w, int h)
1491 appcore_multiwindow_base_class_on_create(instance_h);
1496 EXPORT_API int widget_base_class_on_pause(widget_base_instance_h instance_h)
1498 appcore_multiwindow_base_class_on_pause(instance_h);
1503 EXPORT_API int widget_base_class_on_resume(widget_base_instance_h instance_h)
1505 appcore_multiwindow_base_class_on_resume(instance_h);
1510 EXPORT_API int widget_base_class_on_resize(widget_base_instance_h instance_h,
1516 EXPORT_API int widget_base_class_on_update(widget_base_instance_h instance_h,
1517 bundle *content, int force)
1522 EXPORT_API int widget_base_class_on_destroy(widget_base_instance_h instance_h,
1523 widget_base_destroy_type_e reason, bundle *content)
1525 appcore_multiwindow_base_class_on_terminate(instance_h);
1530 EXPORT_API widget_base_class widget_base_class_get_default(void)
1532 widget_base_class cls;
1534 cls.ops.create = __class_on_create;
1535 cls.ops.resize = __class_on_resize;
1536 cls.ops.update = __class_on_update;
1537 cls.ops.destroy = __class_on_destroy;
1538 cls.ops.pause = __class_on_pause;
1539 cls.ops.resume = __class_on_resume;
1545 EXPORT_API widget_base_class *widget_base_class_add(widget_base_class cls,
1546 const char *class_id, void *class_data)
1548 widget_base_class *c;
1549 appcore_multiwindow_base_class raw_cls;
1551 if (!__is_widget_feature_enabled()) {
1552 LOGE("not supported");
1553 set_last_result(WIDGET_ERROR_NOT_SUPPORTED);
1558 LOGE("class id is NULL");
1559 set_last_result(WIDGET_ERROR_INVALID_PARAMETER);
1563 raw_cls.id = strdup(class_id);
1564 raw_cls.data = class_data;
1565 raw_cls.create = __multiwindow_instance_create;
1566 raw_cls.terminate = __multiwindow_instance_terminate;
1567 raw_cls.pause = __multiwindow_instance_pause;
1568 raw_cls.resume = __multiwindow_instance_resume;
1569 appcore_multiwindow_base_class_add(raw_cls);
1571 c = malloc(sizeof(widget_base_class));
1576 c->id = strdup(class_id);
1577 __context.classes = g_list_append(__context.classes, c);