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>
25 #include <glib-object.h>
28 #include <widget_errno.h>
29 #include <widget_instance.h>
30 #include <aul_app_com.h>
31 #include <Ecore_Wayland.h>
32 #include <system_info.h>
34 #include <vconf-internal-keys.h>
35 #include <screen_connector_provider.h>
36 #include <appcore_multiwindow_base.h>
38 #include "widget_base.h"
44 #define LOG_TAG "CAPI_WIDGET_APPLICATION"
45 #define APP_TYPE_WIDGET "widgetapp"
46 #define STATUS_FOREGROUND "fg"
47 #define STATUS_BACKGROUND "bg"
49 static int __app_event_converter[APPCORE_BASE_EVENT_MAX] = {
50 [APP_EVENT_LOW_MEMORY] = APPCORE_BASE_EVENT_LOW_MEMORY,
51 [APP_EVENT_LOW_BATTERY] = APPCORE_BASE_EVENT_LOW_BATTERY,
52 [APP_EVENT_LANGUAGE_CHANGED] = APPCORE_BASE_EVENT_LANG_CHANGE,
53 [APP_EVENT_DEVICE_ORIENTATION_CHANGED]
54 = APPCORE_BASE_EVENT_DEVICE_ORIENTATION_CHANGED,
55 [APP_EVENT_REGION_FORMAT_CHANGED] = APPCORE_BASE_EVENT_REGION_CHANGE,
56 [APP_EVENT_SUSPENDED_STATE_CHANGED]
57 = APPCORE_BASE_EVENT_SUSPENDED_STATE_CHANGE,
60 struct app_event_info {
61 app_event_type_e type;
65 struct app_event_handler {
66 app_event_type_e type;
72 struct widget_foreach_context {
73 widget_base_instance_cb callback;
77 typedef struct _widget_base_context {
83 } widget_base_context;
85 typedef struct _widget_base_instance_data {
93 char *pending_content;
95 } widget_base_instance_data;
97 static widget_base_context __context;
99 static char *__package_id;
100 static bool __fg_signal;
101 static char *__viewer_endpoint;
102 static bool __is_permanent;
103 static void __call_update_cb(const char *class_id, const char *id, int force,
104 const char *content_raw);
106 static gboolean __timeout_cb(gpointer user_data)
108 widget_base_instance_data *data =
109 (widget_base_instance_data *)user_data;
110 appcore_multiwindow_base_instance_h cxt;
111 const char *class_id;
113 cxt = appcore_multiwindow_base_instance_find(data->id);
114 if (appcore_multiwindow_base_instance_is_resumed(cxt)) {
115 LOGD("Periodic update!");
116 class_id = appcore_multiwindow_base_instance_get_class_id(cxt);
117 __call_update_cb(class_id, data->id, 0, NULL);
119 data->pending_update = true;
120 if (data->periodic_timer) {
121 LOGD("Remove timer!");
122 g_source_remove(data->periodic_timer);
123 data->periodic_timer = 0;
127 return G_SOURCE_CONTINUE;
130 static bool __is_widget_feature_enabled(void)
132 static bool feature = false;
133 static bool retrieved = false;
136 if (retrieved == true)
139 ret = system_info_get_platform_bool(FEATURE_SHELL_APPWIDGET, &feature);
140 if (ret != SYSTEM_INFO_ERROR_NONE) {
141 LOGE("failed to get system info"); /* LCOV_EXCL_LINE */
142 return false; /* LCOV_EXCL_LINE */
150 /* LCOV_EXCL_START */
151 static void __on_poweroff(keynode_t *key, void *data)
155 val = vconf_keynode_get_int(key);
157 case VCONFKEY_SYSMAN_POWER_OFF_DIRECT:
158 case VCONFKEY_SYSMAN_POWER_OFF_RESTART:
159 LOGI("power off changed: %d", val);
162 case VCONFKEY_SYSMAN_POWER_OFF_NONE:
163 case VCONFKEY_SYSMAN_POWER_OFF_POPUP:
171 static void __check_empty_instance(void)
173 int cnt = appcore_multiwindow_base_instance_get_cnt();
179 static void __instance_drop(appcore_multiwindow_base_instance_h instance_h)
181 widget_base_instance_data *data;
183 data = appcore_multiwindow_base_instance_get_extra(instance_h);
184 appcore_multiwindow_base_instance_drop(instance_h);
185 free(data->pending_content);
189 __check_empty_instance();
192 static gint __comp_class(gconstpointer a, gconstpointer b)
194 const widget_base_class *cls = a;
196 return strcmp(cls->id, b);
199 static widget_base_class *__get_class(const char *class_id)
201 widget_base_class *cls;
204 class_node = g_list_find_custom(__context.classes, class_id,
206 if (class_node == NULL) {
207 LOGE("empty classes");
210 cls = (widget_base_class *)class_node->data;
215 static int __send_lifecycle_event(const char *class_id, const char *instance_id,
218 bundle *b = bundle_create();
222 LOGE("out of memory"); /* LCOV_EXCL_LINE */
223 return -1; /* LCOV_EXCL_LINE */
226 bundle_add_str(b, AUL_K_WIDGET_ID, class_id);
227 bundle_add_str(b, AUL_K_WIDGET_INSTANCE_ID, instance_id);
228 bundle_add_byte(b, AUL_K_WIDGET_STATUS, &status, sizeof(int));
229 bundle_add_str(b, AUL_K_PKGID, __package_id);
231 LOGD("send lifecycle %s(%d)", instance_id, status);
232 ret = aul_app_com_send("widget.status", b);
234 LOGE("send lifecycle error:%d", ret); /* LCOV_EXCL_LINE */
241 static int __send_update_status(const char *class_id, const char *instance_id,
242 int status, bundle *extra)
246 bundle_raw *raw = NULL;
251 LOGE("out of memory"); /* LCOV_EXCL_LINE */
252 return -1; /* LCOV_EXCL_LINE */
255 bundle_add_str(b, AUL_K_WIDGET_ID, class_id);
256 bundle_add_str(b, AUL_K_WIDGET_INSTANCE_ID, instance_id);
257 bundle_add_byte(b, AUL_K_WIDGET_STATUS, &status, sizeof(int));
260 bundle_encode(extra, &raw, &len);
261 bundle_add_str(b, WIDGET_K_CONTENT_INFO, (const char *)raw);
262 aul_widget_instance_add(class_id, instance_id);
265 LOGD("send update %s(%d) to %s", instance_id, status, __viewer_endpoint);
266 aul_app_com_send(__viewer_endpoint, b);
269 case WIDGET_INSTANCE_EVENT_CREATE:
270 lifecycle = WIDGET_LIFE_CYCLE_EVENT_CREATE;
272 case WIDGET_INSTANCE_EVENT_DESTROY:
273 lifecycle = WIDGET_LIFE_CYCLE_EVENT_DESTROY;
275 case WIDGET_INSTANCE_EVENT_PAUSE:
276 lifecycle = WIDGET_LIFE_CYCLE_EVENT_PAUSE;
278 case WIDGET_INSTANCE_EVENT_RESUME:
279 lifecycle = WIDGET_LIFE_CYCLE_EVENT_RESUME;
284 __send_lifecycle_event(class_id, instance_id, lifecycle);
293 static void __control_create(const char *class_id, const char *id, bundle *b)
295 widget_base_instance_data *data;
296 char *content = NULL;
298 data = (widget_base_instance_data *)
299 calloc(1, sizeof(widget_base_instance_data));
301 LOGE("Out of memory");
305 data->id = strdup(id);
308 /* call stub create */
309 appcore_multiwindow_base_instance_run(class_id, id, data);
311 bundle_get_str(b, WIDGET_K_CONTENT_INFO, &content);
313 data->content = strdup(content);
317 static void __control_resume(const char *class_id, const char *id, bundle *b)
319 appcore_multiwindow_base_instance_h cxt;
321 cxt = appcore_multiwindow_base_instance_find(id);
323 LOGE("context not found: %s", id);
327 /* call stub resume */
328 appcore_multiwindow_base_instance_resume(cxt);
331 static void __control_pause(const char *class_id, const char *id, bundle *b)
333 appcore_multiwindow_base_instance_h instance_h;
335 instance_h = appcore_multiwindow_base_instance_find(id);
338 LOGE("instance not found: %s", id);
342 /* call stub pause */
343 appcore_multiwindow_base_instance_pause(instance_h);
346 static void __control_resize(const char *class_id, const char *id, bundle *b)
348 appcore_multiwindow_base_instance_h instance_h;
355 widget_base_class *cls;
356 const appcore_multiwindow_base_class *raw_cls;
358 instance_h = appcore_multiwindow_base_instance_find(id);
360 LOGE("context not found: %s", id);
364 raw_cls = appcore_multiwindow_base_instance_get_class(instance_h);
368 cls = __get_class(class_id);
370 LOGE("class not found: %s", class_id);
373 class_data = raw_cls->data;
374 bundle_get_str(b, WIDGET_K_WIDTH, &w_str);
375 bundle_get_str(b, WIDGET_K_HEIGHT, &h_str);
378 w = (int)g_ascii_strtoll(w_str, &remain, 10);
381 h = (int)g_ascii_strtoll(h_str, &remain, 10);
384 cls->ops.resize(instance_h, w, h, class_data);
386 LOGD("%s is resized to %dx%d", id, w, h);
387 __send_update_status(class_id, id,
388 WIDGET_INSTANCE_EVENT_SIZE_CHANGED, NULL);
391 static void __call_update_cb(const char *class_id, const char *id, int force,
392 const char *content_raw)
395 widget_base_class *cls;
396 const appcore_multiwindow_base_class *raw_cls;
397 appcore_multiwindow_base_instance_h instance_h;
398 bundle *content = NULL;
400 instance_h = appcore_multiwindow_base_instance_find(id);
402 LOGE("context not found: %s", id);
406 raw_cls = appcore_multiwindow_base_instance_get_class(instance_h);
408 LOGE("class is NULL");
412 class_data = raw_cls->data;
413 cls = __get_class(class_id);
415 LOGE("class not found: %s", class_id);
419 if (!cls->ops.update) {
420 LOGE("update callback is NULL");
425 content = bundle_decode((const bundle_raw *)content_raw,
426 strlen(content_raw));
430 cls->ops.update(instance_h, content, force, class_data);
432 __send_update_status(class_id, id,
433 WIDGET_INSTANCE_EVENT_UPDATE, NULL);
434 LOGD("updated:%s", id);
437 bundle_free(content);
440 static void __update_pending_content(
441 appcore_multiwindow_base_instance_h instance_h,
442 const char *content_raw)
444 widget_base_instance_data *data;
446 data = (widget_base_instance_data *)
447 appcore_multiwindow_base_instance_get_extra(instance_h);
449 if (data->pending_content) {
450 free(data->pending_content);
451 data->pending_content = NULL;
455 data->pending_content = strdup(content_raw);
456 if (data->pending_content == NULL)
457 LOGW("Out of memory");
460 data->pending_update = true;
463 static void __update_process(const char *class_id, const char *id,
464 appcore_multiwindow_base_instance_h instance_h, void *data)
466 char *content_raw = NULL;
467 char *force_str = NULL;
472 LOGE("bundle is NULL");
476 bundle_get_str(b, WIDGET_K_FORCE, &force_str);
478 if (force_str && strcmp(force_str, "true") == 0)
483 bundle_get_str(b, WIDGET_K_CONTENT_INFO, &content_raw);
484 if (!appcore_multiwindow_base_instance_is_resumed(instance_h) && !force)
485 __update_pending_content(instance_h, content_raw);
487 __call_update_cb(class_id, id, force, content_raw);
490 static void __control_update(const char *class_id, const char *id, bundle *b)
492 appcore_multiwindow_base_instance_h instance_h;
495 appcore_multiwindow_base_instance_foreach(class_id,
496 __update_process, b);
500 instance_h = appcore_multiwindow_base_instance_find(id);
502 LOGE("context not found: %s", id);
506 __update_process(class_id, id, instance_h, b);
509 static void __control_destroy(const char *class_id, const char *id, bundle *b)
511 appcore_multiwindow_base_instance_h instance_h;
512 widget_base_instance_data *data;
514 instance_h = appcore_multiwindow_base_instance_find(id);
516 LOGE("could not find widget obj: %s, clear amd info", id);
517 aul_widget_instance_del(class_id, id);
521 data = (widget_base_instance_data *)
522 appcore_multiwindow_base_instance_get_extra(instance_h);
525 /* call stub terminate */
526 appcore_multiwindow_base_instance_exit(instance_h);
527 free(data->pending_content);
531 __check_empty_instance();
534 static void __control_change_period(const char *class_id, const char *id,
537 appcore_multiwindow_base_instance_h instance_h;
538 widget_base_instance_data *data;
539 double *period = NULL;
543 instance_h = appcore_multiwindow_base_instance_find(id);
545 LOGE("context not found: %s", id);
549 data = (widget_base_instance_data *)
550 appcore_multiwindow_base_instance_get_extra(instance_h);
553 LOGE("could not find instance data: %s", id);
557 if (data->periodic_timer) {
558 LOGD("Remove timer!");
559 g_source_remove(data->periodic_timer);
560 data->periodic_timer = 0;
563 ret = bundle_get_byte(b, WIDGET_K_PERIOD, (void **)&period, &size);
564 if (ret == BUNDLE_ERROR_NONE)
565 data->period = *period;
567 if (data->period > 0) {
568 LOGD("Restart timer!");
569 data->periodic_timer = g_timeout_add_seconds(data->period,
576 static int __multiwindow_create(void *data)
578 char pkgid[256] = {0, };
581 appcore_multiwindow_base_on_create();
582 app_get_id(&__appid);
583 if (aul_app_get_pkgid_bypid(getpid(), pkgid, sizeof(pkgid)) == 0)
584 __package_id = strdup(pkgid);
586 if (!__package_id || !__appid) {
587 LOGE("__package_id is NULL");
591 screen_connector_provider_init();
592 vconf_notify_key_changed(VCONFKEY_SYSMAN_POWER_OFF_STATUS,
593 __on_poweroff, NULL);
596 if (__context.ops.create)
597 ret = __context.ops.create(data);
599 LOGD("widget base is created");
603 static int __multiwindow_terminate(void *data)
605 if (__context.ops.terminate)
606 __context.ops.terminate(data);
608 vconf_ignore_key_changed(VCONFKEY_SYSMAN_POWER_OFF_STATUS,
610 screen_connector_provider_fini();
612 if (__viewer_endpoint) {
613 free(__viewer_endpoint);
614 __viewer_endpoint = NULL;
627 appcore_multiwindow_base_on_terminate();
629 LOGD("widget base is terminated");
633 static int __multiwindow_control(bundle *b, void *data)
635 char *class_id = NULL;
637 char *operation = NULL;
639 appcore_multiwindow_base_on_control(b);
640 bundle_get_str(b, WIDGET_K_CLASS, &class_id);
641 /* for previous version compatibility, use appid for default class id */
642 if (class_id == NULL)
645 bundle_get_str(b, AUL_K_WIDGET_INSTANCE_ID, &id);
646 bundle_get_str(b, WIDGET_K_OPERATION, &operation);
649 LOGE("operation is NULL");
653 if (strcmp(operation, "create") == 0)
654 __control_create(class_id, id, b);
655 else if (strcmp(operation, "resize") == 0)
656 __control_resize(class_id, id, b);
657 else if (strcmp(operation, "update") == 0)
658 __control_update(class_id, id, b);
659 else if (strcmp(operation, "destroy") == 0)
660 __control_destroy(class_id, id, b);
661 else if (strcmp(operation, "resume") == 0)
662 __control_resume(class_id, id, b);
663 else if (strcmp(operation, "pause") == 0)
664 __control_pause(class_id, id, b);
665 else if (strcmp(operation, "terminate") == 0)
666 __control_destroy(class_id, id, b);
667 else if (strcmp(operation, "period") == 0)
668 __control_change_period(class_id, id, b);
673 static void __inst_resume_cb(const char *class_id, const char *id,
674 appcore_multiwindow_base_instance_h cxt, void *data)
676 __control_resume(class_id, id, data);
679 static void __get_content(bundle *b)
681 char *instance_id = NULL;
682 appcore_multiwindow_base_instance_h cxt;
683 widget_base_instance_data * we;
685 bundle_get_str(b, AUL_K_WIDGET_INSTANCE_ID, &instance_id);
687 LOGE("instance id is NULL");
691 cxt = appcore_multiwindow_base_instance_find(instance_id);
693 LOGE("could not find widget obj: %s", instance_id);
697 we = appcore_multiwindow_base_instance_get_extra(cxt);
699 LOGE("widget extra is NULL");
704 bundle_add_str(b, AUL_K_WIDGET_CONTENT_INFO, we->content);
705 LOGD("content info of %s found", instance_id);
707 bundle_add_str(b, AUL_K_WIDGET_CONTENT_INFO, "");
708 LOGD("empty content info added");
712 static int __multiwindow_receive(aul_type type, bundle *b, void *data)
714 appcore_multiwindow_base_on_receive(type, b);
718 appcore_multiwindow_base_instance_foreach_full(
719 __inst_resume_cb, b);
724 case AUL_WIDGET_CONTENT:
734 static void __multiwindow_init(int argc, char **argv, void *data)
736 if (__context.ops.init)
737 __context.ops.init(argc, argv, data);
740 static void __multiwindow_finish(void)
742 if (__context.ops.finish) {
743 __context.ops.finish();
744 /* Check Loader case */
745 if (getenv("AUL_LOADER_INIT")) {
746 unsetenv("AUL_LOADER_INIT");
747 __context.ops.finish();
752 static void __multiwindow_run(void *data)
754 if (__context.ops.run)
755 __context.ops.run(data);
758 static void __multiwindow_exit(void *data)
760 if (__context.ops.exit)
761 __context.ops.exit(data);
764 EXPORT_API int widget_base_exit(void)
766 appcore_multiwindow_base_exit();
767 if (appcore_multiwindow_base_instance_get_cnt() != 0 && __is_permanent)
773 static gboolean __finish_event_cb(gpointer user_data)
775 appcore_multiwindow_base_instance_h cxt = user_data;
778 const char *class_id;
781 LOGE("user_data is NULL");
785 id = appcore_multiwindow_base_instance_get_id(cxt);
786 class_id = appcore_multiwindow_base_instance_get_class_id(cxt);
790 LOGE("Out-of-memory");
794 bundle_add_str(b, WIDGET_K_OPERATION, "terminate");
795 __control_destroy(class_id, id, b);
801 EXPORT_API int widget_base_terminate_context(widget_base_instance_h context)
803 if (!__is_widget_feature_enabled()) {
804 LOGE("not supported"); /* LCOV_EXCL_LINE */
805 return WIDGET_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
809 LOGE("context is null");
810 return WIDGET_ERROR_INVALID_PARAMETER;
813 g_idle_add(__finish_event_cb, context);
815 return WIDGET_ERROR_NONE;
818 static void __inst_full_cb(const char *class_id, const char *id,
819 appcore_multiwindow_base_instance_h cxt, void *data)
821 struct widget_foreach_context *foreach_context = data;
826 if (foreach_context->callback)
827 foreach_context->callback(cxt, foreach_context->data);
830 EXPORT_API int widget_base_foreach_context(widget_base_instance_cb cb, void *data)
832 struct widget_foreach_context foreach_context;
834 if (!__is_widget_feature_enabled()) {
835 LOGE("not supported"); /* LCOV_EXCL_LINE */
836 return WIDGET_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
840 LOGE("callback is NULL");
841 return WIDGET_ERROR_INVALID_PARAMETER;
844 foreach_context.callback = cb;
845 foreach_context.data = data;
846 appcore_multiwindow_base_instance_foreach_full(__inst_full_cb, &foreach_context);
848 return WIDGET_ERROR_NONE;
851 static int __event_cb(void *event, void *data)
853 app_event_handler_h handler = data;
855 struct app_event_info app_event;
857 app_event.type = handler->type;
858 app_event.value = event;
861 handler->cb(&app_event, handler->data);
866 EXPORT_API int widget_base_add_event_handler(app_event_handler_h *event_handler,
867 app_event_type_e event_type,
868 app_event_cb callback,
873 app_event_handler_h handler;
875 r = system_info_get_platform_bool(FEATURE_SHELL_APPWIDGET, &feature);
877 return WIDGET_BASE_ERROR_FAULT;
880 return WIDGET_BASE_ERROR_NOT_SUPPORTED;
882 if (event_handler == NULL || callback == NULL)
883 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
885 if (event_type < APP_EVENT_LOW_MEMORY
886 || event_type > APP_EVENT_REGION_FORMAT_CHANGED)
887 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
889 if (event_type == APP_EVENT_DEVICE_ORIENTATION_CHANGED)
890 return WIDGET_BASE_ERROR_NOT_SUPPORTED;
893 handler = calloc(1, sizeof(struct app_event_handler));
895 return WIDGET_BASE_ERROR_OUT_OF_MEMORY;
897 handler->type = event_type;
898 handler->cb = callback;
899 handler->data = user_data;
900 handler->raw = appcore_base_add_event(
901 __app_event_converter[event_type], __event_cb, handler);
902 *event_handler = handler;
904 return WIDGET_BASE_ERROR_NONE;
907 EXPORT_API int widget_base_remove_event_handler(app_event_handler_h
912 app_event_type_e type;
914 r = system_info_get_platform_bool(FEATURE_SHELL_APPWIDGET, &feature);
916 return WIDGET_BASE_ERROR_FAULT;
919 return WIDGET_BASE_ERROR_NOT_SUPPORTED;
921 if (event_handler == NULL)
922 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
924 type = event_handler->type;
925 if (type < APP_EVENT_LOW_MEMORY ||
926 type > APP_EVENT_REGION_FORMAT_CHANGED)
927 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
929 r = appcore_base_remove_event(event_handler->raw);
931 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
935 return WIDGET_BASE_ERROR_NONE;
938 EXPORT_API int widget_base_context_set_content_info(
939 widget_base_instance_h context,
940 bundle *content_info)
943 bundle_raw *raw = NULL;
946 const char *class_id;
947 widget_base_instance_data *data;
948 appcore_multiwindow_base_instance_h instance_h;
950 if (!__is_widget_feature_enabled()) {
951 LOGE("not supported"); /* LCOV_EXCL_LINE */
952 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
955 if (!context || !content_info)
956 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
958 instance_h = (appcore_multiwindow_base_instance_h)context;
959 id = appcore_multiwindow_base_instance_get_id(instance_h);
960 class_id = appcore_multiwindow_base_instance_get_class_id(instance_h);
961 data = appcore_multiwindow_base_instance_get_extra(instance_h);
963 if (!class_id || !id || !data)
964 return WIDGET_BASE_ERROR_FAULT;
966 ret = __send_update_status(class_id, id,
967 WIDGET_INSTANCE_EVENT_EXTRA_UPDATED, content_info);
972 bundle_encode(content_info, &raw, &len);
974 data->content = strdup((const char *)raw);
976 data->content = NULL;
980 /* LCOV_EXCL_START */
981 LOGE("failed to send content info: %s of %s (%d)", id,
983 return WIDGET_BASE_ERROR_IO_ERROR;
987 return WIDGET_BASE_ERROR_NONE;
990 EXPORT_API int widget_base_context_get_tag(widget_base_instance_h context, void **tag)
992 appcore_multiwindow_base_instance_h instance_h;
993 widget_base_instance_data *data;
995 if (!__is_widget_feature_enabled()) {
996 LOGE("not supported"); /* LCOV_EXCL_LINE */
997 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
1000 if (!context || !tag) {
1001 LOGE("Invalid parameter");
1002 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
1005 instance_h = (appcore_multiwindow_base_instance_h)context;
1006 data = (widget_base_instance_data *)
1007 appcore_multiwindow_base_instance_get_extra(instance_h);
1010 LOGE("Invalid parameter");
1011 return WIDGET_ERROR_INVALID_PARAMETER;
1016 return WIDGET_BASE_ERROR_NONE;
1019 EXPORT_API int widget_base_context_set_tag(widget_base_instance_h context, void *tag)
1021 appcore_multiwindow_base_instance_h instance_h;
1022 widget_base_instance_data *data;
1024 if (!__is_widget_feature_enabled()) {
1025 LOGE("not supported"); /* LCOV_EXCL_LINE */
1026 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
1030 LOGE("Invalid parameter");
1031 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
1034 instance_h = (appcore_multiwindow_base_instance_h)context;
1035 data = (widget_base_instance_data *)
1036 appcore_multiwindow_base_instance_get_extra(instance_h);
1039 return WIDGET_BASE_ERROR_NONE;
1042 EXPORT_API void *widget_base_context_get_user_data(
1043 widget_base_instance_h context)
1045 appcore_multiwindow_base_instance_h instance_h;
1046 widget_base_instance_data *data;
1048 if (!__is_widget_feature_enabled()) {
1049 LOGE("not supported"); /* LCOV_EXCL_LINE */
1050 return NULL; /* LCOV_EXCL_LINE */
1054 LOGE("Invalid parameter");
1058 instance_h = (appcore_multiwindow_base_instance_h)context;
1059 data = (widget_base_instance_data *)
1060 appcore_multiwindow_base_instance_get_extra(instance_h);
1062 return data->user_data;
1066 EXPORT_API int widget_base_context_set_user_data(
1067 widget_base_instance_h context, void *user_data)
1069 appcore_multiwindow_base_instance_h instance_h;
1070 widget_base_instance_data *data;
1072 if (!__is_widget_feature_enabled()) {
1073 LOGE("not supported"); /* LCOV_EXCL_LINE */
1074 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
1078 LOGE("Invalid parameter");
1079 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
1082 instance_h = (appcore_multiwindow_base_instance_h)context;
1083 data = (widget_base_instance_data *)
1084 appcore_multiwindow_base_instance_get_extra(instance_h);
1085 data->user_data = user_data;
1087 return WIDGET_BASE_ERROR_NONE;
1090 EXPORT_API int widget_base_context_get_id(widget_base_instance_h context, char **id)
1092 appcore_multiwindow_base_instance_h instance_h;
1094 if (!__is_widget_feature_enabled()) {
1095 LOGE("not supported"); /* LCOV_EXCL_LINE */
1096 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
1099 instance_h = (appcore_multiwindow_base_instance_h)context;
1100 *id = (char *)appcore_multiwindow_base_instance_get_id(instance_h);
1102 return WIDGET_BASE_ERROR_NONE;
1105 EXPORT_API const char *widget_base_get_viewer_endpoint()
1107 return __viewer_endpoint;
1110 EXPORT_API int widget_base_init(widget_base_ops ops, int argc, char **argv,
1114 char *viewer_endpoint = NULL;
1115 appcore_multiwindow_base_ops raw_ops
1116 = appcore_multiwindow_base_get_default_ops();
1118 __context.ops = ops;
1119 __context.argc = argc;
1120 __context.argv = argv;
1121 __context.data = data;
1123 /* override methods */
1124 raw_ops.base.create = __multiwindow_create;
1125 raw_ops.base.control = __multiwindow_control;
1126 raw_ops.base.terminate = __multiwindow_terminate;
1127 raw_ops.base.receive = __multiwindow_receive;
1128 raw_ops.base.init = __multiwindow_init;
1129 raw_ops.base.finish = __multiwindow_finish;
1130 raw_ops.base.run = __multiwindow_run;
1131 raw_ops.base.exit = __multiwindow_exit;
1133 if (!__is_widget_feature_enabled()) {
1134 LOGE("not supported"); /* LCOV_EXCL_LINE */
1135 return WIDGET_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
1138 kb = bundle_import_from_argv(argc, argv);
1140 bundle_get_str(kb, WIDGET_K_ENDPOINT, &viewer_endpoint);
1141 if (viewer_endpoint) {
1142 LOGD("viewer endpoint :%s", viewer_endpoint);
1143 __viewer_endpoint = strdup(viewer_endpoint);
1145 LOGE("endpoint is missing");
1150 LOGE("failed to get launch argv"); /* LCOV_EXCL_LINE */
1151 return WIDGET_ERROR_FAULT;
1154 if (appcore_multiwindow_base_init(raw_ops, argc, argv, data) < 0)
1155 return WIDGET_ERROR_FAULT;
1157 return WIDGET_ERROR_NONE;
1160 static int __on_create(void *data)
1162 return widget_base_on_create();
1165 static int __on_terminate(void *data)
1167 return widget_base_on_terminate();
1170 static void __on_init(int argc, char **argv, void *data)
1172 widget_base_on_init(argc, argv);
1175 static void __on_finish(void)
1177 widget_base_on_finish();
1180 static void __on_run(void *data)
1182 widget_base_on_run();
1185 static void __on_exit(void *data)
1187 widget_base_on_exit();
1190 EXPORT_API int widget_base_on_create(void)
1192 appcore_multiwindow_base_on_create();
1197 EXPORT_API int widget_base_on_terminate(void)
1199 appcore_multiwindow_base_on_terminate();
1204 EXPORT_API int widget_base_on_init(int argc, char **argv)
1209 EXPORT_API void widget_base_on_finish(void)
1213 EXPORT_API void widget_base_on_run(void)
1217 EXPORT_API void widget_base_on_exit(void)
1221 EXPORT_API widget_base_ops widget_base_get_default_ops(void)
1223 widget_base_ops ops;
1225 /* override methods */
1226 ops.create = __on_create;
1227 ops.terminate = __on_terminate;
1228 ops.init = __on_init;
1229 ops.finish = __on_finish;
1231 ops.exit = __on_exit;
1236 static void __free_class(gpointer data)
1238 widget_base_class *cls = data;
1244 EXPORT_API void widget_base_fini(void)
1246 g_list_free_full(__context.classes, __free_class);
1247 __context.classes = NULL;
1249 appcore_multiwindow_base_fini();
1252 EXPORT_API int widget_base_context_window_bind(
1253 widget_base_instance_h instance_h, const char *id,
1254 Ecore_Wl_Window *wl_win)
1256 struct wl_surface *surface;
1258 surface = ecore_wl_window_surface_get(wl_win);
1259 if (surface == NULL) {
1260 LOGE("failed to get surface"); /* LCOV_EXCL_LINE */
1261 return WIDGET_BASE_ERROR_FAULT; /* LCOV_EXCL_LINE */
1264 screen_connector_provider_remote_enable(id, surface);
1265 appcore_multiwindow_base_window_bind(instance_h, wl_win);
1267 return WIDGET_BASE_ERROR_NONE;
1270 static int __class_on_create(widget_base_instance_h instance_h, bundle *content,
1271 int w, int h, void *class_data)
1273 return widget_base_class_on_create(instance_h, content, w, h);
1276 static int __class_on_resume(widget_base_instance_h instance_h, void *class_data)
1278 return widget_base_class_on_resume(instance_h);
1281 static int __class_on_pause(widget_base_instance_h instance_h,
1284 return widget_base_class_on_pause(instance_h);
1287 static int __class_on_resize(widget_base_instance_h instance_h, int w, int h,
1290 return widget_base_class_on_resize(instance_h, w, h);
1293 static int __class_on_update(widget_base_instance_h instance_h, bundle *content,
1294 int force, void *class_data)
1296 return widget_base_class_on_update(instance_h, content, force);
1299 static int __class_on_destroy(widget_base_instance_h instance_h,
1300 widget_base_destroy_type_e reason, bundle *content,
1303 return widget_base_class_on_destroy(instance_h, reason, content);
1306 static void __multiwindow_instance_create(
1307 appcore_multiwindow_base_instance_h instance_h,
1310 widget_base_instance_data *instance_data;
1312 bundle *content_info = NULL;
1314 char *class_id = NULL;
1315 char *operation = NULL;
1316 char *content = NULL;
1319 char *remain = NULL;
1323 widget_base_class *cls;
1324 double *period = NULL;
1327 appcore_multiwindow_base_class_on_create(instance_h);
1328 instance_data = appcore_multiwindow_base_instance_get_extra(instance_h);
1329 b = instance_data->args;
1331 bundle_get_str(b, WIDGET_K_CLASS, &class_id);
1332 /* for previous version compatibility, use appid for default class id */
1333 if (class_id == NULL)
1336 cls = __get_class(class_id);
1338 LOGE("class not found: %s", class_id);
1342 bundle_get_str(b, AUL_K_WIDGET_INSTANCE_ID, &id);
1343 bundle_get_str(b, WIDGET_K_OPERATION, &operation);
1346 LOGE("no operation provided");
1350 bundle_get_str(b, WIDGET_K_CONTENT_INFO, &content);
1351 bundle_get_str(b, WIDGET_K_WIDTH, &w_str);
1352 bundle_get_str(b, WIDGET_K_HEIGHT, &h_str);
1355 w = (int)g_ascii_strtoll(w_str, &remain, 10);
1358 h = (int)g_ascii_strtoll(h_str, &remain, 10);
1361 content_info = bundle_decode((const bundle_raw *)content,
1364 if (cls->ops.create)
1365 ret = cls->ops.create(instance_h, content_info, w, h, class_data);
1368 LOGW("Create callback returns error(%d)", ret);
1369 ret = __send_update_status(class_id, id,
1370 WIDGET_INSTANCE_EVENT_CREATE_ABORTED, NULL);
1372 LOGE("Fail to send abort status (%d) ", ret);
1373 __instance_drop(instance_h);
1375 LOGD("%s is created", id);
1376 ret = __send_update_status(class_id, id,
1377 WIDGET_INSTANCE_EVENT_CREATE, NULL);
1379 LOGE("Fail to send create status (%d) ", ret);
1381 aul_widget_instance_add(class_id, id);
1383 ret = bundle_get_byte(b, WIDGET_K_PERIOD, (void **)&period,
1385 if (ret == BUNDLE_ERROR_NONE) {
1386 instance_data->period = *period;
1387 instance_data->periodic_timer = g_timeout_add_seconds(
1388 instance_data->period,
1389 __timeout_cb, instance_data);
1394 bundle_free(content_info);
1397 static void __multiwindow_instance_resume(
1398 appcore_multiwindow_base_instance_h instance_h,
1402 const char *class_id;
1403 widget_base_class *cls;
1404 widget_base_instance_data *data;
1406 appcore_multiwindow_base_class_on_resume(instance_h);
1407 id = appcore_multiwindow_base_instance_get_id(instance_h);
1408 class_id = appcore_multiwindow_base_instance_get_class_id(instance_h);
1409 cls = __get_class(class_id);
1411 LOGE("class not found: %s", class_id);
1415 data = (widget_base_instance_data *)
1416 appcore_multiwindow_base_instance_get_extra(instance_h);
1418 if (data->pending_update) {
1419 LOGD("pending update!");
1420 data->pending_update = false;
1421 __call_update_cb(class_id, data->id, 0, data->pending_content);
1422 if (data->period > 0) {
1423 LOGD("Restart timer!");
1424 data->periodic_timer = g_timeout_add_seconds(
1426 __timeout_cb, data);
1430 if (cls->ops.resume)
1431 cls->ops.resume(instance_h, class_data);
1433 LOGD("%s is resumed", id);
1434 __send_update_status(class_id, id,
1435 WIDGET_INSTANCE_EVENT_RESUME, NULL);
1438 LOGD("Send fg signal to resourceD");
1439 aul_send_app_status_change_signal(getpid(),
1448 static void __multiwindow_instance_pause(
1449 appcore_multiwindow_base_instance_h instance_h,
1453 const char *class_id;
1454 widget_base_class *cls;
1456 appcore_multiwindow_base_class_on_pause(instance_h);
1457 id = appcore_multiwindow_base_instance_get_id(instance_h);
1458 class_id = appcore_multiwindow_base_instance_get_class_id(instance_h);
1459 cls = __get_class(class_id);
1461 LOGE("class not found: %s", class_id);
1466 cls->ops.pause(instance_h, class_data);
1468 LOGD("%s is paused", id);
1469 __send_update_status(class_id, id,
1470 WIDGET_INSTANCE_EVENT_PAUSE, NULL);
1473 LOGD("Send bg signal to resourceD");
1474 aul_send_app_status_change_signal(getpid(),
1479 __fg_signal = false;
1483 static void __multiwindow_instance_terminate(
1484 appcore_multiwindow_base_instance_h instance_h,
1487 widget_base_instance_data *data;
1489 char *operation = NULL;
1490 bundle *content_info;
1491 widget_base_destroy_type_e reason = WIDGET_BASE_DESTROY_TYPE_TEMPORARY;
1492 int event = WIDGET_INSTANCE_EVENT_TERMINATE;
1494 const char *class_id;
1495 widget_base_class *cls;
1497 id = appcore_multiwindow_base_instance_get_id(instance_h);
1498 class_id = appcore_multiwindow_base_instance_get_class_id(instance_h);
1499 data = appcore_multiwindow_base_instance_get_extra(
1500 (appcore_multiwindow_base_instance_h)instance_h);
1502 cls = __get_class(class_id);
1504 LOGE("class not found: %s", class_id);
1509 bundle_get_str(b, WIDGET_K_OPERATION, &operation);
1510 if (operation && strcmp(operation, "destroy") == 0)
1511 reason = WIDGET_BASE_DESTROY_TYPE_PERMANENT;
1515 content_info = bundle_decode((const bundle_raw *)data->content,
1516 strlen(data->content));
1518 content_info = bundle_create();
1520 if (cls->ops.destroy)
1521 cls->ops.destroy(instance_h, reason, content_info, class_data);
1523 LOGD("%s is destroyed %d", id, reason);
1524 if (reason == WIDGET_BASE_DESTROY_TYPE_PERMANENT) {
1525 __is_permanent = true;
1526 event = WIDGET_INSTANCE_EVENT_DESTROY;
1527 aul_widget_instance_del(class_id, id);
1529 __is_permanent = false;
1530 __send_update_status(class_id, id,
1531 WIDGET_INSTANCE_EVENT_EXTRA_UPDATED,
1536 bundle_free(content_info);
1538 if (data->periodic_timer)
1539 g_source_remove(data->periodic_timer);
1541 __send_update_status(class_id, id, event, NULL);
1542 appcore_multiwindow_base_class_on_terminate(instance_h);
1545 EXPORT_API int widget_base_class_on_create(widget_base_instance_h instance_h,
1546 bundle *content, int w, int h)
1548 appcore_multiwindow_base_class_on_create(instance_h);
1553 EXPORT_API int widget_base_class_on_pause(widget_base_instance_h instance_h)
1555 appcore_multiwindow_base_class_on_pause(instance_h);
1560 EXPORT_API int widget_base_class_on_resume(widget_base_instance_h instance_h)
1562 appcore_multiwindow_base_class_on_resume(instance_h);
1567 EXPORT_API int widget_base_class_on_resize(widget_base_instance_h instance_h,
1573 EXPORT_API int widget_base_class_on_update(widget_base_instance_h instance_h,
1574 bundle *content, int force)
1579 EXPORT_API int widget_base_class_on_destroy(widget_base_instance_h instance_h,
1580 widget_base_destroy_type_e reason, bundle *content)
1582 appcore_multiwindow_base_class_on_terminate(instance_h);
1587 EXPORT_API widget_base_class widget_base_class_get_default(void)
1589 widget_base_class cls;
1591 cls.ops.create = __class_on_create;
1592 cls.ops.resize = __class_on_resize;
1593 cls.ops.update = __class_on_update;
1594 cls.ops.destroy = __class_on_destroy;
1595 cls.ops.pause = __class_on_pause;
1596 cls.ops.resume = __class_on_resume;
1601 EXPORT_API widget_base_class *widget_base_class_add(widget_base_class cls,
1602 const char *class_id, void *class_data)
1604 widget_base_class *c;
1605 appcore_multiwindow_base_class raw_cls;
1607 if (!__is_widget_feature_enabled()) {
1608 LOGE("not supported");
1609 set_last_result(WIDGET_ERROR_NOT_SUPPORTED);
1614 LOGE("class is is NULL");
1615 set_last_result(WIDGET_ERROR_INVALID_PARAMETER);
1619 raw_cls.id = strdup(class_id);
1620 raw_cls.data = class_data;
1621 raw_cls.create = __multiwindow_instance_create;
1622 raw_cls.terminate = __multiwindow_instance_terminate;
1623 raw_cls.pause = __multiwindow_instance_pause;
1624 raw_cls.resume = __multiwindow_instance_resume;
1625 appcore_multiwindow_base_class_add(raw_cls);
1627 c = malloc(sizeof(widget_base_class));
1632 c->id = strdup(class_id);
1633 __context.classes = g_list_append(__context.classes, c);