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 {
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);
113 if (appcore_multiwindow_base_instance_is_resumed(cxt)) {
114 LOGD("Periodic update!");
115 class_id = appcore_multiwindow_base_instance_get_class_id(cxt);
116 __call_update_cb(class_id, data->id, 0, NULL);
118 data->pending_update = true;
119 if (data->periodic_timer) {
120 LOGD("Remove timer!");
121 g_source_remove(data->periodic_timer);
122 data->periodic_timer = 0;
126 return G_SOURCE_CONTINUE;
129 static bool __is_widget_feature_enabled(void)
131 static bool feature = false;
132 static bool retrieved = false;
135 if (retrieved == true)
138 ret = system_info_get_platform_bool(FEATURE_SHELL_APPWIDGET, &feature);
139 if (ret != SYSTEM_INFO_ERROR_NONE) {
140 LOGE("failed to get system info"); /* LCOV_EXCL_LINE */
141 return false; /* LCOV_EXCL_LINE */
149 /* LCOV_EXCL_START */
150 static void __on_poweroff(keynode_t *key, void *data)
154 val = vconf_keynode_get_int(key);
156 case VCONFKEY_SYSMAN_POWER_OFF_DIRECT:
157 case VCONFKEY_SYSMAN_POWER_OFF_RESTART:
158 LOGI("power off changed: %d", val);
161 case VCONFKEY_SYSMAN_POWER_OFF_NONE:
162 case VCONFKEY_SYSMAN_POWER_OFF_POPUP:
170 static void __check_empty_instance(void)
172 int cnt = appcore_multiwindow_base_instance_get_cnt();
178 static void __instance_drop(appcore_multiwindow_base_instance_h instance_h)
180 widget_base_instance_data *data;
182 data = appcore_multiwindow_base_instance_get_extra(instance_h);
183 appcore_multiwindow_base_instance_drop(instance_h);
187 __check_empty_instance();
190 static gint __comp_class(gconstpointer a, gconstpointer b)
192 const widget_base_class *cls = a;
194 return strcmp(cls->id, b);
197 static widget_base_class *__get_class(const char *class_id)
199 widget_base_class *cls;
202 class_node = g_list_find_custom(__context.classes, class_id,
204 if (class_node == NULL) {
205 LOGE("empty classes");
208 cls = (widget_base_class *)class_node->data;
213 static int __send_lifecycle_event(const char *class_id, const char *instance_id,
216 bundle *b = bundle_create();
220 LOGE("out of memory"); /* LCOV_EXCL_LINE */
221 return -1; /* LCOV_EXCL_LINE */
224 bundle_add_str(b, AUL_K_WIDGET_ID, class_id);
225 bundle_add_str(b, AUL_K_WIDGET_INSTANCE_ID, instance_id);
226 bundle_add_byte(b, AUL_K_WIDGET_STATUS, &status, sizeof(int));
227 bundle_add_str(b, AUL_K_PKGID, __package_id);
229 LOGD("send lifecycle %s(%d)", instance_id, status);
230 ret = aul_app_com_send("widget.status", b);
232 LOGE("send lifecycle error:%d", ret); /* LCOV_EXCL_LINE */
239 static int __send_update_status(const char *class_id, const char *instance_id,
240 int status, bundle *extra)
244 bundle_raw *raw = NULL;
249 LOGE("out of memory"); /* LCOV_EXCL_LINE */
250 return -1; /* LCOV_EXCL_LINE */
253 bundle_add_str(b, AUL_K_WIDGET_ID, class_id);
254 bundle_add_str(b, AUL_K_WIDGET_INSTANCE_ID, instance_id);
255 bundle_add_byte(b, AUL_K_WIDGET_STATUS, &status, sizeof(int));
258 bundle_encode(extra, &raw, &len);
259 bundle_add_str(b, WIDGET_K_CONTENT_INFO, (const char *)raw);
260 aul_widget_instance_add(class_id, instance_id);
263 LOGD("send update %s(%d) to %s", instance_id, status, __viewer_endpoint);
264 aul_app_com_send(__viewer_endpoint, b);
267 case WIDGET_INSTANCE_EVENT_CREATE:
268 lifecycle = WIDGET_LIFE_CYCLE_EVENT_CREATE;
270 case WIDGET_INSTANCE_EVENT_DESTROY:
271 lifecycle = WIDGET_LIFE_CYCLE_EVENT_DESTROY;
273 case WIDGET_INSTANCE_EVENT_PAUSE:
274 lifecycle = WIDGET_LIFE_CYCLE_EVENT_PAUSE;
276 case WIDGET_INSTANCE_EVENT_RESUME:
277 lifecycle = WIDGET_LIFE_CYCLE_EVENT_RESUME;
282 __send_lifecycle_event(class_id, instance_id, lifecycle);
291 static void __control_create(const char *class_id, const char *id, bundle *b)
293 widget_base_instance_data *data;
294 char *content = NULL;
295 double *period = NULL;
299 data = (widget_base_instance_data *)
300 calloc(1, sizeof(widget_base_instance_data));
302 LOGE("Out of memory");
306 data->id = strdup(id);
308 ret = bundle_get_byte(b, WIDGET_K_PERIOD, (void **)&period, &size);
309 if (ret == BUNDLE_ERROR_NONE) {
310 data->period = *period;
311 data->periodic_timer = g_timeout_add_seconds(data->period,
315 /* call stub create */
316 appcore_multiwindow_base_instance_run(class_id, id, data);
318 bundle_get_str(b, WIDGET_K_CONTENT_INFO, &content);
320 data->content = strdup(content);
324 static void __control_resume(const char *class_id, const char *id, bundle *b)
326 appcore_multiwindow_base_instance_h cxt;
328 cxt = appcore_multiwindow_base_instance_find(id);
330 LOGE("context not found: %s", id);
334 /* call stub resume */
335 appcore_multiwindow_base_instance_resume(cxt);
338 static void __control_pause(const char *class_id, const char *id, bundle *b)
340 appcore_multiwindow_base_instance_h instance_h;
342 instance_h = appcore_multiwindow_base_instance_find(id);
345 LOGE("instance not found: %s", id);
349 /* call stub pause */
350 appcore_multiwindow_base_instance_pause(instance_h);
353 static void __control_resize(const char *class_id, const char *id, bundle *b)
355 appcore_multiwindow_base_instance_h instance_h;
362 widget_base_class *cls;
363 const appcore_multiwindow_base_class *raw_cls;
365 instance_h = appcore_multiwindow_base_instance_find(id);
367 LOGE("context not found: %s", id);
371 raw_cls = appcore_multiwindow_base_instance_get_class(instance_h);
375 cls = __get_class(class_id);
377 LOGE("class not found: %s", class_id);
380 class_data = raw_cls->data;
381 bundle_get_str(b, WIDGET_K_WIDTH, &w_str);
382 bundle_get_str(b, WIDGET_K_HEIGHT, &h_str);
385 w = (int)g_ascii_strtoll(w_str, &remain, 10);
388 h = (int)g_ascii_strtoll(h_str, &remain, 10);
391 cls->ops.resize(instance_h, w, h, class_data);
393 LOGD("%s is resized to %dx%d", id, w, h);
394 __send_update_status(class_id, id,
395 WIDGET_INSTANCE_EVENT_SIZE_CHANGED, NULL);
398 static void __call_update_cb(const char *class_id, const char *id, int force,
399 const char *content_raw)
402 widget_base_class *cls;
403 const appcore_multiwindow_base_class *raw_cls;
404 appcore_multiwindow_base_instance_h instance_h;
405 bundle *content = NULL;
407 instance_h = appcore_multiwindow_base_instance_find(id);
409 LOGE("context not found: %s", id);
413 raw_cls = appcore_multiwindow_base_instance_get_class(instance_h);
415 LOGE("class is NULL");
419 class_data = raw_cls->data;
420 cls = __get_class(class_id);
422 LOGE("class not found: %s", class_id);
426 if (!cls->ops.update) {
427 LOGE("update callback is NULL");
432 content = bundle_decode((const bundle_raw *)content_raw,
433 strlen(content_raw));
437 cls->ops.update(instance_h, content, force, class_data);
439 __send_update_status(class_id, id,
440 WIDGET_INSTANCE_EVENT_UPDATE, NULL);
441 LOGD("updated:%s", id);
444 bundle_free(content);
447 static void __update_process(const char *class_id, const char *id,
448 appcore_multiwindow_base_instance_h instance_h, void *data)
450 char *content_raw = NULL;
451 char *force_str = NULL;
456 LOGE("bundle is NULL");
460 bundle_get_str(b, WIDGET_K_FORCE, &force_str);
462 if (force_str && strcmp(force_str, "true") == 0)
467 bundle_get_str(b, WIDGET_K_CONTENT_INFO, &content_raw);
468 __call_update_cb(class_id, id, force, content_raw);
471 static void __control_update(const char *class_id, const char *id, bundle *b)
473 appcore_multiwindow_base_instance_h instance_h;
476 appcore_multiwindow_base_instance_foreach(class_id,
477 __update_process, b);
481 instance_h = appcore_multiwindow_base_instance_find(id);
483 LOGE("context not found: %s", id);
487 __update_process(class_id, id, instance_h, b);
490 static void __control_destroy(const char *class_id, const char *id, bundle *b)
492 appcore_multiwindow_base_instance_h instance_h;
493 widget_base_instance_data *data;
495 instance_h = appcore_multiwindow_base_instance_find(id);
497 LOGE("could not find widget obj: %s, clear amd info", id);
498 aul_widget_instance_del(class_id, id);
502 data = (widget_base_instance_data *)
503 appcore_multiwindow_base_instance_get_extra(instance_h);
506 /* call stub terminate */
507 appcore_multiwindow_base_instance_exit(instance_h);
511 __check_empty_instance();
514 static void __control_change_period(const char *class_id, const char *id,
517 appcore_multiwindow_base_instance_h instance_h;
518 widget_base_instance_data *data;
519 double *period = NULL;
523 instance_h = appcore_multiwindow_base_instance_find(id);
525 LOGE("context not found: %s", id);
529 data = (widget_base_instance_data *)
530 appcore_multiwindow_base_instance_get_extra(instance_h);
533 LOGE("could not find instance data: %s", id);
537 if (data->periodic_timer) {
538 LOGD("Remove timer!");
539 g_source_remove(data->periodic_timer);
540 data->periodic_timer = 0;
543 ret = bundle_get_byte(b, WIDGET_K_PERIOD, (void **)&period, &size);
544 if (ret == BUNDLE_ERROR_NONE)
545 data->period = *period;
547 if (data->period > 0) {
548 LOGD("Restart timer!");
549 data->periodic_timer = g_timeout_add_seconds(data->period,
556 static int __multiwindow_create(void *data)
558 char pkgid[256] = {0, };
561 appcore_multiwindow_base_on_create();
562 app_get_id(&__appid);
563 if (aul_app_get_pkgid_bypid(getpid(), pkgid, sizeof(pkgid)) == 0)
564 __package_id = strdup(pkgid);
566 if (!__package_id || !__appid) {
567 LOGE("__package_id is NULL");
571 screen_connector_provider_init();
572 vconf_notify_key_changed(VCONFKEY_SYSMAN_POWER_OFF_STATUS,
573 __on_poweroff, NULL);
576 if (__context.ops.create)
577 ret = __context.ops.create(data);
579 LOGD("widget base is created");
583 static int __multiwindow_terminate(void *data)
585 if (__context.ops.terminate)
586 __context.ops.terminate(data);
588 vconf_ignore_key_changed(VCONFKEY_SYSMAN_POWER_OFF_STATUS,
590 screen_connector_provider_fini();
592 if (__viewer_endpoint) {
593 free(__viewer_endpoint);
594 __viewer_endpoint = NULL;
607 appcore_multiwindow_base_on_terminate();
609 LOGD("widget base is terminated");
613 static int __multiwindow_control(bundle *b, void *data)
615 char *class_id = NULL;
617 char *operation = NULL;
619 appcore_multiwindow_base_on_control(b);
620 bundle_get_str(b, WIDGET_K_CLASS, &class_id);
621 /* for previous version compatibility, use appid for default class id */
622 if (class_id == NULL)
625 bundle_get_str(b, AUL_K_WIDGET_INSTANCE_ID, &id);
626 bundle_get_str(b, WIDGET_K_OPERATION, &operation);
629 LOGE("operation is NULL");
633 if (strcmp(operation, "create") == 0)
634 __control_create(class_id, id, b);
635 else if (strcmp(operation, "resize") == 0)
636 __control_resize(class_id, id, b);
637 else if (strcmp(operation, "update") == 0)
638 __control_update(class_id, id, b);
639 else if (strcmp(operation, "destroy") == 0)
640 __control_destroy(class_id, id, b);
641 else if (strcmp(operation, "resume") == 0)
642 __control_resume(class_id, id, b);
643 else if (strcmp(operation, "pause") == 0)
644 __control_pause(class_id, id, b);
645 else if (strcmp(operation, "terminate") == 0)
646 __control_destroy(class_id, id, b);
647 else if (strcmp(operation, "period") == 0)
648 __control_change_period(class_id, id, b);
653 static void __inst_resume_cb(const char *class_id, const char *id,
654 appcore_multiwindow_base_instance_h cxt, void *data)
656 __control_resume(class_id, id, data);
659 static void __get_content(bundle *b)
661 char *instance_id = NULL;
662 appcore_multiwindow_base_instance_h cxt;
663 widget_base_instance_data * we;
665 bundle_get_str(b, AUL_K_WIDGET_INSTANCE_ID, &instance_id);
667 LOGE("instance id is NULL");
671 cxt = appcore_multiwindow_base_instance_find(instance_id);
673 LOGE("could not find widget obj: %s", instance_id);
677 we = appcore_multiwindow_base_instance_get_extra(cxt);
679 LOGE("widget extra is NULL");
684 bundle_add_str(b, AUL_K_WIDGET_CONTENT_INFO, we->content);
685 LOGD("content info of %s found", instance_id);
687 bundle_add_str(b, AUL_K_WIDGET_CONTENT_INFO, "");
688 LOGD("empty content info added");
692 static int __multiwindow_receive(aul_type type, bundle *b, void *data)
694 appcore_multiwindow_base_on_receive(type, b);
698 appcore_multiwindow_base_instance_foreach_full(
699 __inst_resume_cb, b);
704 case AUL_WIDGET_CONTENT:
714 static void __multiwindow_init(int argc, char **argv, void *data)
716 if (__context.ops.init)
717 __context.ops.init(argc, argv, data);
720 static void __multiwindow_finish(void)
722 if (__context.ops.finish) {
723 __context.ops.finish();
724 /* Check Loader case */
725 if (getenv("AUL_LOADER_INIT")) {
726 unsetenv("AUL_LOADER_INIT");
727 __context.ops.finish();
732 static void __multiwindow_run(void *data)
734 if (__context.ops.run)
735 __context.ops.run(data);
738 static void __multiwindow_exit(void *data)
740 if (__context.ops.exit)
741 __context.ops.exit(data);
744 EXPORT_API int widget_base_exit(void)
746 appcore_multiwindow_base_exit();
747 if (appcore_multiwindow_base_instance_get_cnt() != 0 && __is_permanent)
753 static gboolean __finish_event_cb(gpointer user_data)
755 appcore_multiwindow_base_instance_h cxt = user_data;
758 const char *class_id;
761 LOGE("user_data is NULL");
765 id = appcore_multiwindow_base_instance_get_id(cxt);
766 class_id = appcore_multiwindow_base_instance_get_class_id(cxt);
770 LOGE("Out-of-memory");
774 bundle_add_str(b, WIDGET_K_OPERATION, "terminate");
775 __control_destroy(class_id, id, b);
781 EXPORT_API int widget_base_terminate_context(widget_base_instance_h context)
783 if (!__is_widget_feature_enabled()) {
784 LOGE("not supported"); /* LCOV_EXCL_LINE */
785 return WIDGET_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
789 LOGE("context is null");
790 return WIDGET_ERROR_INVALID_PARAMETER;
793 g_idle_add(__finish_event_cb, context);
795 return WIDGET_ERROR_NONE;
798 static void __inst_full_cb(const char *class_id, const char *id,
799 appcore_multiwindow_base_instance_h cxt, void *data)
801 struct widget_foreach_context *foreach_context = data;
806 if (foreach_context->callback)
807 foreach_context->callback(cxt, foreach_context->data);
810 EXPORT_API int widget_base_foreach_context(widget_base_instance_cb cb, void *data)
812 struct widget_foreach_context foreach_context;
814 if (!__is_widget_feature_enabled()) {
815 LOGE("not supported"); /* LCOV_EXCL_LINE */
816 return WIDGET_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
820 LOGE("callback is NULL");
821 return WIDGET_ERROR_INVALID_PARAMETER;
824 foreach_context.callback = cb;
825 foreach_context.data = data;
826 appcore_multiwindow_base_instance_foreach_full(__inst_full_cb, &foreach_context);
828 return WIDGET_ERROR_NONE;
831 static int __event_cb(void *event, void *data)
833 app_event_handler_h handler = data;
835 struct app_event_info app_event;
837 app_event.type = handler->type;
838 app_event.value = event;
841 handler->cb(&app_event, handler->data);
846 EXPORT_API int widget_base_add_event_handler(app_event_handler_h *event_handler,
847 app_event_type_e event_type,
848 app_event_cb callback,
853 app_event_handler_h handler;
855 r = system_info_get_platform_bool(FEATURE_SHELL_APPWIDGET, &feature);
857 return WIDGET_BASE_ERROR_FAULT;
860 return WIDGET_BASE_ERROR_NOT_SUPPORTED;
862 if (event_handler == NULL || callback == NULL)
863 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
865 if (event_type < APP_EVENT_LOW_MEMORY
866 || event_type > APP_EVENT_REGION_FORMAT_CHANGED)
867 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
869 if (event_type == APP_EVENT_DEVICE_ORIENTATION_CHANGED)
870 return WIDGET_BASE_ERROR_NOT_SUPPORTED;
873 handler = calloc(1, sizeof(struct app_event_handler));
875 return WIDGET_BASE_ERROR_OUT_OF_MEMORY;
877 handler->type = event_type;
878 handler->cb = callback;
879 handler->data = user_data;
880 handler->raw = appcore_base_add_event(
881 __app_event_converter[event_type], __event_cb, handler);
882 *event_handler = handler;
884 return WIDGET_BASE_ERROR_NONE;
887 EXPORT_API int widget_base_remove_event_handler(app_event_handler_h
892 app_event_type_e type;
894 r = system_info_get_platform_bool(FEATURE_SHELL_APPWIDGET, &feature);
896 return WIDGET_BASE_ERROR_FAULT;
899 return WIDGET_BASE_ERROR_NOT_SUPPORTED;
901 if (event_handler == NULL)
902 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
904 type = event_handler->type;
905 if (type < APP_EVENT_LOW_MEMORY ||
906 type > APP_EVENT_REGION_FORMAT_CHANGED)
907 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
909 r = appcore_base_remove_event(event_handler->raw);
911 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
915 return WIDGET_BASE_ERROR_NONE;
918 EXPORT_API int widget_base_context_set_content_info(
919 widget_base_instance_h context,
920 bundle *content_info)
923 bundle_raw *raw = NULL;
926 const char *class_id;
927 widget_base_instance_data *data;
928 appcore_multiwindow_base_instance_h instance_h;
930 if (!__is_widget_feature_enabled()) {
931 LOGE("not supported"); /* LCOV_EXCL_LINE */
932 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
935 if (!context || !content_info)
936 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
938 instance_h = (appcore_multiwindow_base_instance_h)context;
939 id = appcore_multiwindow_base_instance_get_id(instance_h);
940 class_id = appcore_multiwindow_base_instance_get_class_id(instance_h);
941 data = appcore_multiwindow_base_instance_get_extra(instance_h);
943 if (!class_id || !id || !data)
944 return WIDGET_BASE_ERROR_FAULT;
946 ret = __send_update_status(class_id, id,
947 WIDGET_INSTANCE_EVENT_EXTRA_UPDATED, content_info);
952 bundle_encode(content_info, &raw, &len);
954 data->content = strdup((const char *)raw);
956 data->content = NULL;
960 /* LCOV_EXCL_START */
961 LOGE("failed to send content info: %s of %s (%d)", id,
963 return WIDGET_BASE_ERROR_IO_ERROR;
967 return WIDGET_BASE_ERROR_NONE;
970 EXPORT_API int widget_base_context_get_tag(widget_base_instance_h context, void **tag)
972 appcore_multiwindow_base_instance_h instance_h;
973 widget_base_instance_data *data;
975 if (!__is_widget_feature_enabled()) {
976 LOGE("not supported"); /* LCOV_EXCL_LINE */
977 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
980 if (!context || !tag) {
981 LOGE("Invalid parameter");
982 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
985 instance_h = (appcore_multiwindow_base_instance_h)context;
986 data = (widget_base_instance_data *)
987 appcore_multiwindow_base_instance_get_extra(instance_h);
990 LOGE("Invalid parameter");
991 return WIDGET_ERROR_INVALID_PARAMETER;
996 return WIDGET_BASE_ERROR_NONE;
999 EXPORT_API int widget_base_context_set_tag(widget_base_instance_h context, void *tag)
1001 appcore_multiwindow_base_instance_h instance_h;
1002 widget_base_instance_data *data;
1004 if (!__is_widget_feature_enabled()) {
1005 LOGE("not supported"); /* LCOV_EXCL_LINE */
1006 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
1010 LOGE("Invalid parameter");
1011 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
1014 instance_h = (appcore_multiwindow_base_instance_h)context;
1015 data = (widget_base_instance_data *)
1016 appcore_multiwindow_base_instance_get_extra(instance_h);
1019 return WIDGET_BASE_ERROR_NONE;
1022 EXPORT_API void *widget_base_context_get_user_data(
1023 widget_base_instance_h context)
1025 appcore_multiwindow_base_instance_h instance_h;
1026 widget_base_instance_data *data;
1028 if (!__is_widget_feature_enabled()) {
1029 LOGE("not supported"); /* LCOV_EXCL_LINE */
1030 return NULL; /* LCOV_EXCL_LINE */
1034 LOGE("Invalid parameter");
1038 instance_h = (appcore_multiwindow_base_instance_h)context;
1039 data = (widget_base_instance_data *)
1040 appcore_multiwindow_base_instance_get_extra(instance_h);
1042 return data->user_data;
1046 EXPORT_API int widget_base_context_set_user_data(
1047 widget_base_instance_h context, void *user_data)
1049 appcore_multiwindow_base_instance_h instance_h;
1050 widget_base_instance_data *data;
1052 if (!__is_widget_feature_enabled()) {
1053 LOGE("not supported"); /* LCOV_EXCL_LINE */
1054 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
1058 LOGE("Invalid parameter");
1059 return WIDGET_BASE_ERROR_INVALID_PARAMETER;
1062 instance_h = (appcore_multiwindow_base_instance_h)context;
1063 data = (widget_base_instance_data *)
1064 appcore_multiwindow_base_instance_get_extra(instance_h);
1065 data->user_data = user_data;
1067 return WIDGET_BASE_ERROR_NONE;
1070 EXPORT_API int widget_base_context_get_id(widget_base_instance_h context, char **id)
1072 appcore_multiwindow_base_instance_h instance_h;
1074 if (!__is_widget_feature_enabled()) {
1075 LOGE("not supported"); /* LCOV_EXCL_LINE */
1076 return WIDGET_BASE_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
1079 instance_h = (appcore_multiwindow_base_instance_h)context;
1080 *id = (char *)appcore_multiwindow_base_instance_get_id(instance_h);
1082 return WIDGET_BASE_ERROR_NONE;
1085 EXPORT_API const char *widget_base_get_viewer_endpoint()
1087 return __viewer_endpoint;
1090 EXPORT_API int widget_base_init(widget_base_ops ops, int argc, char **argv,
1094 char *viewer_endpoint = NULL;
1095 appcore_multiwindow_base_ops raw_ops
1096 = appcore_multiwindow_base_get_default_ops();
1098 __context.ops = ops;
1099 __context.argc = argc;
1100 __context.argv = argv;
1101 __context.data = data;
1103 /* override methods */
1104 raw_ops.base.create = __multiwindow_create;
1105 raw_ops.base.control = __multiwindow_control;
1106 raw_ops.base.terminate = __multiwindow_terminate;
1107 raw_ops.base.receive = __multiwindow_receive;
1108 raw_ops.base.init = __multiwindow_init;
1109 raw_ops.base.finish = __multiwindow_finish;
1110 raw_ops.base.run = __multiwindow_run;
1111 raw_ops.base.exit = __multiwindow_exit;
1113 if (!__is_widget_feature_enabled()) {
1114 LOGE("not supported"); /* LCOV_EXCL_LINE */
1115 return WIDGET_ERROR_NOT_SUPPORTED; /* LCOV_EXCL_LINE */
1118 kb = bundle_import_from_argv(argc, argv);
1120 bundle_get_str(kb, WIDGET_K_ENDPOINT, &viewer_endpoint);
1121 if (viewer_endpoint) {
1122 LOGD("viewer endpoint :%s", viewer_endpoint);
1123 __viewer_endpoint = strdup(viewer_endpoint);
1125 LOGE("endpoint is missing");
1130 LOGE("failed to get launch argv"); /* LCOV_EXCL_LINE */
1131 return WIDGET_ERROR_FAULT;
1134 if (appcore_multiwindow_base_init(raw_ops, argc, argv, data) < 0)
1135 return WIDGET_ERROR_FAULT;
1137 return WIDGET_ERROR_NONE;
1140 static int __on_create(void *data)
1142 return widget_base_on_create();
1145 static int __on_terminate(void *data)
1147 return widget_base_on_terminate();
1150 static void __on_init(int argc, char **argv, void *data)
1152 widget_base_on_init(argc, argv);
1155 static void __on_finish(void)
1157 widget_base_on_finish();
1160 static void __on_run(void *data)
1162 widget_base_on_run();
1165 static void __on_exit(void *data)
1167 widget_base_on_exit();
1170 EXPORT_API int widget_base_on_create(void)
1172 appcore_multiwindow_base_on_create();
1177 EXPORT_API int widget_base_on_terminate(void)
1179 appcore_multiwindow_base_on_terminate();
1184 EXPORT_API int widget_base_on_init(int argc, char **argv)
1189 EXPORT_API void widget_base_on_finish(void)
1193 EXPORT_API void widget_base_on_run(void)
1197 EXPORT_API void widget_base_on_exit(void)
1201 EXPORT_API widget_base_ops widget_base_get_default_ops(void)
1203 widget_base_ops ops;
1205 /* override methods */
1206 ops.create = __on_create;
1207 ops.terminate = __on_terminate;
1208 ops.init = __on_init;
1209 ops.finish = __on_finish;
1211 ops.exit = __on_exit;
1216 static void __free_class(gpointer data)
1218 widget_base_class *cls = data;
1224 EXPORT_API void widget_base_fini(void)
1226 g_list_free_full(__context.classes, __free_class);
1227 __context.classes = NULL;
1229 appcore_multiwindow_base_fini();
1232 EXPORT_API int widget_base_context_window_bind(
1233 widget_base_instance_h instance_h, const char *id,
1234 Ecore_Wl_Window *wl_win)
1236 struct wl_surface *surface;
1238 surface = ecore_wl_window_surface_get(wl_win);
1239 if (surface == NULL) {
1240 LOGE("failed to get surface"); /* LCOV_EXCL_LINE */
1241 return WIDGET_BASE_ERROR_FAULT; /* LCOV_EXCL_LINE */
1244 screen_connector_provider_remote_enable(id, surface);
1245 appcore_multiwindow_base_window_bind(instance_h, wl_win);
1247 return WIDGET_BASE_ERROR_NONE;
1250 static int __class_on_create(widget_base_instance_h instance_h, bundle *content,
1251 int w, int h, void *class_data)
1253 return widget_base_class_on_create(instance_h, content, w, h);
1256 static int __class_on_resume(widget_base_instance_h instance_h, void *class_data)
1258 return widget_base_class_on_resume(instance_h);
1261 static int __class_on_pause(widget_base_instance_h instance_h,
1264 return widget_base_class_on_pause(instance_h);
1267 static int __class_on_resize(widget_base_instance_h instance_h, int w, int h,
1270 return widget_base_class_on_resize(instance_h, w, h);
1273 static int __class_on_update(widget_base_instance_h instance_h, bundle *content,
1274 int force, void *class_data)
1276 return widget_base_class_on_update(instance_h, content, force);
1279 static int __class_on_destroy(widget_base_instance_h instance_h,
1280 widget_base_destroy_type_e reason, bundle *content,
1283 return widget_base_class_on_destroy(instance_h, reason, content);
1286 static void __multiwindow_instance_create(
1287 appcore_multiwindow_base_instance_h instance_h,
1290 widget_base_instance_data *instance_data;
1292 bundle *content_info = NULL;
1294 char *class_id = NULL;
1295 char *operation = NULL;
1296 char *content = NULL;
1299 char *remain = NULL;
1303 widget_base_class *cls;
1305 appcore_multiwindow_base_class_on_create(instance_h);
1306 instance_data = appcore_multiwindow_base_instance_get_extra(instance_h);
1307 b = instance_data->args;
1309 bundle_get_str(b, WIDGET_K_CLASS, &class_id);
1310 /* for previous version compatibility, use appid for default class id */
1311 if (class_id == NULL)
1314 cls = __get_class(class_id);
1316 LOGE("class not found: %s", class_id);
1320 bundle_get_str(b, AUL_K_WIDGET_INSTANCE_ID, &id);
1321 bundle_get_str(b, WIDGET_K_OPERATION, &operation);
1324 LOGE("no operation provided");
1328 bundle_get_str(b, WIDGET_K_CONTENT_INFO, &content);
1329 bundle_get_str(b, WIDGET_K_WIDTH, &w_str);
1330 bundle_get_str(b, WIDGET_K_HEIGHT, &h_str);
1333 w = (int)g_ascii_strtoll(w_str, &remain, 10);
1336 h = (int)g_ascii_strtoll(h_str, &remain, 10);
1339 content_info = bundle_decode((const bundle_raw *)content,
1342 if (cls->ops.create)
1343 ret = cls->ops.create(instance_h, content_info, w, h, class_data);
1346 LOGW("Create callback returns error(%d)", ret);
1347 ret = __send_update_status(class_id, id,
1348 WIDGET_INSTANCE_EVENT_CREATE_ABORTED, NULL);
1349 __instance_drop(instance_h);
1351 LOGD("%s is created", id);
1352 ret = __send_update_status(class_id, id,
1353 WIDGET_INSTANCE_EVENT_CREATE, NULL);
1355 aul_widget_instance_add(class_id, id);
1359 bundle_free(content_info);
1362 static void __multiwindow_instance_resume(
1363 appcore_multiwindow_base_instance_h instance_h,
1367 const char *class_id;
1368 widget_base_class *cls;
1369 widget_base_instance_data *data;
1371 appcore_multiwindow_base_class_on_resume(instance_h);
1372 id = appcore_multiwindow_base_instance_get_id(instance_h);
1373 class_id = appcore_multiwindow_base_instance_get_class_id(instance_h);
1374 cls = __get_class(class_id);
1376 LOGE("class not found: %s", class_id);
1380 data = (widget_base_instance_data *)
1381 appcore_multiwindow_base_instance_get_extra(instance_h);
1383 if (data->pending_update) {
1384 LOGD("pending update!");
1385 data->pending_update = false;
1386 __control_update(class_id, data->id, data->args);
1387 if (data->period > 0) {
1388 LOGD("Restart timer!");
1389 data->periodic_timer = g_timeout_add_seconds(
1391 __timeout_cb, data);
1395 if (cls->ops.resume)
1396 cls->ops.resume(instance_h, class_data);
1398 LOGD("%s is resumed", id);
1399 __send_update_status(class_id, id,
1400 WIDGET_INSTANCE_EVENT_RESUME, NULL);
1403 LOGD("Send fg signal to resourceD");
1404 aul_send_app_status_change_signal(getpid(),
1413 static void __multiwindow_instance_pause(
1414 appcore_multiwindow_base_instance_h instance_h,
1418 const char *class_id;
1419 widget_base_class *cls;
1421 appcore_multiwindow_base_class_on_pause(instance_h);
1422 id = appcore_multiwindow_base_instance_get_id(instance_h);
1423 class_id = appcore_multiwindow_base_instance_get_class_id(instance_h);
1424 cls = __get_class(class_id);
1426 LOGE("class not found: %s", class_id);
1431 cls->ops.pause(instance_h, class_data);
1433 LOGD("%s is paused", id);
1434 __send_update_status(class_id, id,
1435 WIDGET_INSTANCE_EVENT_PAUSE, NULL);
1438 LOGD("Send bg signal to resourceD");
1439 aul_send_app_status_change_signal(getpid(),
1444 __fg_signal = false;
1448 static void __multiwindow_instance_terminate(
1449 appcore_multiwindow_base_instance_h instance_h,
1452 widget_base_instance_data *data;
1454 char *operation = NULL;
1455 bundle *content_info;
1456 widget_base_destroy_type_e reason = WIDGET_BASE_DESTROY_TYPE_TEMPORARY;
1457 int event = WIDGET_INSTANCE_EVENT_TERMINATE;
1459 const char *class_id;
1460 widget_base_class *cls;
1462 id = appcore_multiwindow_base_instance_get_id(instance_h);
1463 class_id = appcore_multiwindow_base_instance_get_class_id(instance_h);
1464 data = appcore_multiwindow_base_instance_get_extra(
1465 (appcore_multiwindow_base_instance_h)instance_h);
1467 cls = __get_class(class_id);
1469 LOGE("class not found: %s", class_id);
1474 bundle_get_str(b, WIDGET_K_OPERATION, &operation);
1475 if (operation && strcmp(operation, "destroy") == 0)
1476 reason = WIDGET_BASE_DESTROY_TYPE_PERMANENT;
1480 content_info = bundle_decode((const bundle_raw *)data->content,
1481 strlen(data->content));
1483 content_info = bundle_create();
1485 if (cls->ops.destroy)
1486 cls->ops.destroy(instance_h, reason, content_info, class_data);
1488 LOGD("%s is destroyed %d", id, reason);
1489 if (reason == WIDGET_BASE_DESTROY_TYPE_PERMANENT) {
1490 __is_permanent = true;
1491 event = WIDGET_INSTANCE_EVENT_DESTROY;
1492 aul_widget_instance_del(class_id, id);
1494 __is_permanent = false;
1495 __send_update_status(class_id, id,
1496 WIDGET_INSTANCE_EVENT_EXTRA_UPDATED,
1501 bundle_free(content_info);
1503 if (data->periodic_timer)
1504 g_source_remove(data->periodic_timer);
1506 __send_update_status(class_id, id, event, NULL);
1507 appcore_multiwindow_base_class_on_terminate(instance_h);
1510 EXPORT_API int widget_base_class_on_create(widget_base_instance_h instance_h,
1511 bundle *content, int w, int h)
1513 appcore_multiwindow_base_class_on_create(instance_h);
1518 EXPORT_API int widget_base_class_on_pause(widget_base_instance_h instance_h)
1520 appcore_multiwindow_base_class_on_pause(instance_h);
1525 EXPORT_API int widget_base_class_on_resume(widget_base_instance_h instance_h)
1527 appcore_multiwindow_base_class_on_resume(instance_h);
1532 EXPORT_API int widget_base_class_on_resize(widget_base_instance_h instance_h,
1538 EXPORT_API int widget_base_class_on_update(widget_base_instance_h instance_h,
1539 bundle *content, int force)
1544 EXPORT_API int widget_base_class_on_destroy(widget_base_instance_h instance_h,
1545 widget_base_destroy_type_e reason, bundle *content)
1547 appcore_multiwindow_base_class_on_terminate(instance_h);
1552 EXPORT_API widget_base_class widget_base_class_get_default(void)
1554 widget_base_class cls;
1556 cls.ops.create = __class_on_create;
1557 cls.ops.resize = __class_on_resize;
1558 cls.ops.update = __class_on_update;
1559 cls.ops.destroy = __class_on_destroy;
1560 cls.ops.pause = __class_on_pause;
1561 cls.ops.resume = __class_on_resume;
1566 EXPORT_API widget_base_class *widget_base_class_add(widget_base_class cls,
1567 const char *class_id, void *class_data)
1569 widget_base_class *c;
1570 appcore_multiwindow_base_class raw_cls;
1572 if (!__is_widget_feature_enabled()) {
1573 LOGE("not supported");
1574 set_last_result(WIDGET_ERROR_NOT_SUPPORTED);
1579 LOGE("class is is NULL");
1580 set_last_result(WIDGET_ERROR_INVALID_PARAMETER);
1584 raw_cls.id = strdup(class_id);
1585 raw_cls.data = class_data;
1586 raw_cls.create = __multiwindow_instance_create;
1587 raw_cls.terminate = __multiwindow_instance_terminate;
1588 raw_cls.pause = __multiwindow_instance_pause;
1589 raw_cls.resume = __multiwindow_instance_resume;
1590 appcore_multiwindow_base_class_add(raw_cls);
1592 c = malloc(sizeof(widget_base_class));
1597 c->id = strdup(class_id);
1598 __context.classes = g_list_append(__context.classes, c);