2 * Copyright (c) 2016 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.
17 #include <sys/types.h>
18 #include <sys/socket.h>
26 #include <linux/limits.h>
28 #include <Ecore_Wayland.h>
29 #include <wayland-client.h>
30 #include <wayland-tbm-client.h>
31 #include <tizen-extension-client-protocol.h>
34 #include <Ecore_Evas.h>
35 #include <Ecore_Input_Evas.h>
36 #include <Elementary.h>
37 #include <glib-object.h>
44 #include <bundle_internal.h>
46 #include <voice_control_elm.h>
47 #include <voice_control_elm_private.h>
49 #include "appcore_base.h"
50 #include "appcore_ui_base.h"
51 #include "appcore_ui_base_private.h"
61 typedef struct _appcore_ui_base_context {
62 appcore_ui_base_ops ops;
71 Ecore_Event_Handler *hshow;
72 Ecore_Event_Handler *hhide;
73 Ecore_Event_Handler *hvchange;
74 Ecore_Event_Handler *hlower;
75 } appcore_ui_base_context;
78 static bool b_active = false;
79 static bool first_launch = true;
87 static GSList *g_winnode_list;
88 static appcore_ui_base_context __context;
89 static struct wl_display *dsp;
90 static struct wl_registry *reg;
91 static struct tizen_policy *tz_policy;
92 static bool bg_state = false;
94 static void __wl_listener_cb(void *data, struct wl_registry *reg,
95 uint32_t id, const char *interface, uint32_t ver)
97 if (interface && !strcmp(interface, "tizen_policy")) {
99 tz_policy = wl_registry_bind(reg, id,
100 &tizen_policy_interface, 1);
104 static void __wl_listener_remove_cb(void *data, struct wl_registry *reg,
110 static const struct wl_registry_listener reg_listener = {
112 __wl_listener_remove_cb
115 static Eina_Bool __stub_show_cb(void *data, int type, void *event)
117 if (__context.ops.window.show)
118 __context.ops.window.show(type, event, __context.data);
120 return ECORE_CALLBACK_RENEW;
123 static Eina_Bool __stub_hide_cb(void *data, int type, void *event)
125 if (__context.ops.window.hide)
126 __context.ops.window.hide(type, event, __context.data);
128 return ECORE_CALLBACK_RENEW;
131 static Eina_Bool __stub_visibility_cb(void *data, int type, void *event)
133 if (__context.ops.window.visibility)
134 __context.ops.window.visibility(type, event, __context.data);
136 return ECORE_CALLBACK_RENEW;
139 static Eina_Bool __stub_lower_cb(void *data, int type, void *event)
141 if (__context.ops.window.lower)
142 __context.ops.window.lower(type, event, __context.data);
144 return ECORE_CALLBACK_RENEW;
147 static void __prepare_to_suspend(void)
149 int suspend = APPCORE_BASE_SUSPENDED_STATE_WILL_ENTER_SUSPEND;
151 if (appcore_base_is_bg_allowed() && !appcore_base_is_suspended()) {
152 appcore_base_raise_event((void *)&suspend, APPCORE_BASE_EVENT_SUSPENDED_STATE_CHANGE);
153 appcore_base_toggle_suspended_state();
157 static void __exit_from_suspend(void)
159 int suspend = APPCORE_BASE_SUSPENDED_STATE_DID_EXIT_FROM_SUSPEND;
161 if (appcore_base_is_suspended()) {
162 appcore_base_raise_event((void *)&suspend, APPCORE_BASE_EVENT_SUSPENDED_STATE_CHANGE);
163 appcore_base_toggle_suspended_state();
167 static void __do_pause(void)
169 if (__context.state == AS_RUNNING) {
170 if (__context.ops.pause)
171 __context.ops.pause(__context.data);
173 __context.state = AS_PAUSED;
174 __prepare_to_suspend();
176 aul_status_update(STATUS_BG);
179 static void __do_resume(void)
181 if (__context.state == AS_PAUSED || __context.state == AS_CREATED) {
182 __exit_from_suspend();
183 if (__context.ops.resume) {
184 LOG(LOG_DEBUG, "LAUNCH", "[%s:Application:resume:start]", __context.appid);
185 __context.ops.resume(__context.data);
186 LOG(LOG_DEBUG, "LAUNCH", "[%s:Application:resume:done]", __context.appid);
187 LOG(LOG_DEBUG, "LAUNCH", "[%s:Application:Launching:done]", __context.appid);
189 if ((__context.hint & APPCORE_UI_BASE_HINT_WINDOW_STACK_CONTROL) &&
190 __context.below_app) {
191 aul_app_group_activate_below(__context.below_app);
192 free(__context.below_app);
193 __context.below_app = NULL;
195 __context.state = AS_RUNNING;
198 aul_status_update(STATUS_VISIBLE);
201 static GSList *__find_win(unsigned int win)
206 for (iter = g_winnode_list; iter; iter = g_slist_next(iter)) {
208 if (t && t->win == win)
215 static int __get_main_window(void)
217 struct win_node *entry = NULL;
219 if (g_winnode_list != NULL) {
220 entry = g_winnode_list->data;
221 return (unsigned int) entry->win;
227 static int __get_main_surface(void)
229 struct win_node *entry = NULL;
231 if (g_winnode_list != NULL) {
232 entry = g_winnode_list->data;
233 return (unsigned int) entry->surf;
239 static bool __add_win(unsigned int win, unsigned int surf)
244 _DBG("[EVENT_TEST][EVENT] __add_win WIN:%x\n", win);
249 _DBG("[EVENT_TEST][EVENT] ERROR There is already window : %x \n", win);
253 t = calloc(1, sizeof(struct win_node));
259 t->bfobscured = FALSE;
261 g_winnode_list = g_slist_append(g_winnode_list, t);
266 static bool __delete_win(unsigned int win)
273 _DBG("[EVENT_TEST][EVENT] ERROR There is no window : %x \n",
279 g_winnode_list = g_slist_delete_link(g_winnode_list, f);
284 static bool __update_win(unsigned int win, unsigned int surf, bool bfobscured)
289 _DBG("[EVENT_TEST][EVENT] __update_win WIN:%x fully_obscured %d\n", win,
295 _DBG("[EVENT_TEST][EVENT] ERROR There is no window : %x \n", win);
299 t = (struct win_node *)f->data;
303 t->bfobscured = bfobscured;
308 static void __raise_win(void)
310 Ecore_Wl_Window *win;
313 if (!(__context.hint & APPCORE_UI_BASE_HINT_WINDOW_STACK_CONTROL))
316 win_id = __get_main_window();
318 _DBG("Raise window: %d", win_id);
319 win = ecore_wl_window_find(win_id);
320 ecore_wl_window_activate(win);
323 static void __pause_win(void)
325 Ecore_Wl_Window *win;
326 GSList *wlist = g_winnode_list;
327 struct win_node *entry = NULL;
329 if (!(__context.hint & APPCORE_UI_BASE_HINT_WINDOW_STACK_CONTROL))
332 _DBG("Pause window");
337 _DBG("Pause window: %d", entry->win);
338 win = ecore_wl_window_find(entry->win);
339 ecore_wl_window_iconified_set(win, EINA_TRUE);
345 static int __init_wl(void)
347 _DBG("initialize wayland");
348 dsp = wl_display_connect(NULL);
350 _ERR("Failed to connect wl display");
354 reg = wl_display_get_registry(dsp);
356 _ERR("Failed to get registry");
357 wl_display_disconnect(dsp);
361 wl_registry_add_listener(reg, ®_listener, NULL);
362 wl_display_roundtrip(dsp);
365 _ERR("Failed to get tizen policy interface");
366 wl_registry_destroy(reg);
367 wl_display_disconnect(dsp);
374 static void __finish_wl(void)
377 tizen_policy_destroy(tz_policy);
382 wl_registry_destroy(reg);
387 wl_display_disconnect(dsp);
392 static void __set_bg_state(void)
394 if (!tz_policy && __init_wl() < 0)
397 tizen_policy_set_background_state(tz_policy, getpid());
398 wl_display_roundtrip(dsp);
400 _DBG("bg state: %d", bg_state);
403 static void __unset_bg_state(void)
408 tizen_policy_unset_background_state(tz_policy, getpid());
409 wl_display_roundtrip(dsp);
411 _DBG("bg state: %d", bg_state);
414 static void __do_start(bundle *b)
416 const char *bg_launch;
417 const char *below_app;
419 if (__context.hint & APPCORE_UI_BASE_HINT_WINDOW_STACK_CONTROL) {
420 if (__context.below_app) {
421 free(__context.below_app);
422 __context.below_app = NULL;
425 below_app = bundle_get_val(b, AUL_SVC_K_RELOCATE_BELOW);
427 __context.below_app = strdup(below_app);
431 first_launch = FALSE;
435 if (__context.hint & APPCORE_UI_BASE_HINT_BG_LAUNCH_CONTROL) {
436 bg_launch = bundle_get_val(b, AUL_SVC_K_BG_LAUNCH);
437 if (bg_launch && strcmp(bg_launch, "enable") == 0) {
438 if (!bg_state && __context.state != AS_RUNNING)
450 EXPORT_API int appcore_ui_base_on_receive(aul_type type, bundle *b)
452 if (__context.state == AS_DYING) {
453 _ERR("Skip the event in dying state");
457 if (type == AUL_TERMINATE_BGAPP && __context.state != AS_PAUSED)
460 if (type == AUL_START)
461 __exit_from_suspend();
463 appcore_base_on_receive(type, b);
476 case AUL_TERMINATE_BGAPP:
477 _DBG("[APP %d] is paused. TERMINATE", getpid());
478 __context.state = AS_DYING;
479 aul_status_update(STATUS_DYING);
480 if (__context.ops.base.exit)
481 __context.ops.base.exit(__context.data);
493 EXPORT_API int appcore_ui_base_on_create(void)
495 appcore_base_on_create();
497 __context.hshow = ecore_event_handler_add(ECORE_WL_EVENT_WINDOW_SHOW, __stub_show_cb, NULL);
498 __context.hhide = ecore_event_handler_add(ECORE_WL_EVENT_WINDOW_HIDE, __stub_hide_cb, NULL);
499 __context.hvchange = ecore_event_handler_add(ECORE_WL_EVENT_WINDOW_VISIBILITY_CHANGE,
500 __stub_visibility_cb, NULL);
501 __context.hlower = ecore_event_handler_add(ECORE_WL_EVENT_WINDOW_LOWER, __stub_lower_cb, NULL);
502 __context.state = AS_CREATED;
503 LOG(LOG_DEBUG, "LAUNCH", "[%s:Application:create:done]", __context.appid);
508 EXPORT_API int appcore_ui_base_on_terminate(void)
510 if (__context.state == AS_RUNNING) {
511 if (__context.ops.pause)
512 __context.ops.pause(__context.data);
515 __context.state = AS_DYING;
517 appcore_base_on_terminate();
522 EXPORT_API int appcore_ui_base_on_pause(void)
527 EXPORT_API int appcore_ui_base_on_resume(void)
532 EXPORT_API int appcore_ui_base_on_control(bundle *b)
534 LOG(LOG_DEBUG, "LAUNCH", "[%s:Application:reset:start]", __context.appid);
535 appcore_base_on_control(b);
536 LOG(LOG_DEBUG, "LAUNCH", "[%s:Application:reset:done]", __context.appid);
541 static void __group_attach()
543 if (!(__context.hint & APPCORE_UI_BASE_HINT_WINDOW_GROUP_CONTROL))
546 appcore_ui_base_group_add();
549 static void __group_lower()
551 if (!(__context.hint & APPCORE_UI_BASE_HINT_WINDOW_GROUP_CONTROL))
553 appcore_ui_base_group_remove();
556 EXPORT_API void appcore_ui_base_group_add()
558 static bool attached = false;
560 _DBG("__group_attach");
564 int wid = __get_main_surface();
566 _ERR("window wasn't ready");
570 aul_app_group_set_window(wid);
574 EXPORT_API void appcore_ui_base_group_remove()
578 _DBG("__group_lower");
579 aul_app_group_lower(&exit);
581 _DBG("__group_lower : sub-app!");
582 if (__context.ops.base.exit)
583 __context.ops.base.exit(__context.data);
587 EXPORT_API void appcore_ui_base_window_on_show(int type, void *event)
589 Ecore_Wl_Event_Window_Show *ev;
592 if (ev->parent_win != 0) {
593 /* This is child window. Skip!!! */
597 _DBG("[EVENT_TEST][EVENT] GET SHOW EVENT!!!. WIN:%x, %d\n", ev->win, ev->data[0]);
599 if (!__find_win((unsigned int)ev->win))
600 __add_win((unsigned int)ev->win, (unsigned int)ev->data[0]);
602 __update_win((unsigned int)ev->win, (unsigned int)ev->data[0], FALSE);
604 if (ev->data[0] != 0)
608 static bool __check_visible(void)
611 struct win_node *entry = NULL;
613 _DBG("[EVENT_TEST][EVENT] __check_visible\n");
615 for (iter = g_winnode_list; iter != NULL; iter = g_slist_next(iter)) {
617 _DBG("win : %x obscured : %d\n", entry->win, entry->bfobscured);
618 if (entry->bfobscured == FALSE)
625 EXPORT_API void appcore_ui_base_window_on_hide(int type, void *event)
627 Ecore_Wl_Event_Window_Hide *ev;
631 _DBG("[EVENT_TEST][EVENT] GET HIDE EVENT!!!. WIN:%x\n", ev->win);
633 if (__find_win((unsigned int)ev->win)) {
634 __delete_win((unsigned int)ev->win);
635 bvisibility = __check_visible();
636 if (!bvisibility && b_active == TRUE) {
637 _DBG(" Go to Pasue state \n");
644 EXPORT_API void appcore_ui_base_window_on_lower(int type, void *event)
646 Ecore_Wl_Event_Window_Lower *ev;
651 _DBG("ECORE_WL_EVENT_WINDOW_LOWER window id:%u\n", ev->win);
655 EXPORT_API void appcore_ui_base_window_on_visibility(int type, void *event)
657 Ecore_Wl_Event_Window_Visibility_Change *ev;
661 __update_win((unsigned int)ev->win, 0, ev->fully_obscured);
662 bvisibility = __check_visible();
664 _DBG("bvisibility %d, b_active %d", bvisibility, b_active);
666 if (bvisibility && b_active == FALSE) {
667 _DBG(" Go to Resume state\n");
670 } else if (!bvisibility && b_active == TRUE) {
671 _DBG(" Go to Pasue state \n");
675 _DBG(" No change state \n");
680 EXPORT_API int appcore_ui_base_init(appcore_ui_base_ops ops, int argc, char **argv,
681 void *data, unsigned int hint)
684 const char *bg_launch;
686 char appid[PATH_MAX] = {0, };
688 bool is_vc_vt_automode = false;
690 aul_app_get_appid_bypid(getpid(), appid, sizeof(appid));
692 __context.data = data;
693 __context.argc = argc;
694 __context.argv = argv;
695 __context.hint = hint;
696 __context.state = AS_NONE;
697 __context.appid = strdup(appid);
699 LOG(LOG_DEBUG, "LAUNCH", "[%s:Application:main:done]", appid);
700 elm_init(argc, argv);
702 r = vc_elm_is_supported_vt_auto(&is_vc_vt_automode);
704 _ERR("[VC] Fail to get vconfkey of vt_automode");
706 if (is_vc_vt_automode == true) {
708 vc_elm_set_auto_register_mode(2, 0);
712 if (__context.hint & APPCORE_UI_BASE_HINT_HW_ACC_CONTROL) {
713 hwacc = getenv("HWACC");
716 _DBG("elm_config_accel_preference_set is not called");
717 } else if (strcmp(hwacc, "USE") == 0) {
718 elm_config_accel_preference_set("hw");
719 _DBG("elm_config_accel_preference_set : hw");
720 } else if (strcmp(hwacc, "NOT_USE") == 0) {
721 elm_config_accel_preference_set("none");
722 _DBG("elm_config_accel_preference_set : none");
724 _DBG("elm_config_accel_preference_set is not called");
728 if (__context.hint & APPCORE_UI_BASE_HINT_BG_LAUNCH_CONTROL) {
729 b = bundle_import_from_argv(argc, argv);
731 bg_launch = bundle_get_val(b, AUL_SVC_K_BG_LAUNCH);
732 if (bg_launch && strcmp(bg_launch, "enable") == 0)
739 return appcore_base_init(ops.base, argc, argv, data);
742 EXPORT_API void appcore_ui_base_fini(void)
744 bool is_vc_vt_automode = false;
750 ecore_event_handler_del(__context.hshow);
752 ecore_event_handler_del(__context.hhide);
753 if (__context.hvchange)
754 ecore_event_handler_del(__context.hvchange);
755 if (__context.hlower)
756 ecore_event_handler_del(__context.hlower);
760 r = vc_elm_is_supported_vt_auto(&is_vc_vt_automode);
762 _ERR("[VC] Fail to get vconfkey of vt_automode");
764 if (is_vc_vt_automode == true)
765 vc_elm_deinitialize();
770 /* Check loader case */
771 if (getenv("AUL_LOADER_INIT")) {
772 unsetenv("AUL_LOADER_INIT");
776 free(__context.appid);
777 __context.appid = NULL;
780 EXPORT_API void appcore_ui_base_pause(void)
785 EXPORT_API void appcore_ui_base_resume(void)
790 EXPORT_API bool appcore_ui_base_is_resumed(void)
792 return __context.state == AS_RUNNING;
795 EXPORT_API void appcore_ui_base_exit(void)
797 if (__context.ops.base.exit)
798 __context.ops.base.exit(__context.data);
801 EXPORT_API unsigned int appcore_ui_base_get_main_window(void)
803 return __get_main_window();
806 EXPORT_API unsigned int appcore_ui_base_get_main_surface(void)
808 return __get_main_surface();
811 static int __on_receive(aul_type type, bundle *b, void *data)
813 return appcore_ui_base_on_receive(type, b);
816 static int __on_create(void *data)
818 return appcore_ui_base_on_create();
821 static int __on_terminate(void *data)
823 return appcore_ui_base_on_terminate();
826 static int __on_pause(void *data)
828 return appcore_ui_base_on_pause();
831 static int __on_resume(void *data)
833 return appcore_ui_base_on_resume();
836 static void __window_on_show(int type, void *event, void *data)
838 appcore_ui_base_window_on_show(type, event);
841 static void __window_on_hide(int type, void *event, void *data)
843 appcore_ui_base_window_on_hide(type, event);
846 static void __window_on_lower(int type, void *event, void *data)
848 appcore_ui_base_window_on_lower(type, event);
851 static void __window_on_visibility(int type, void *event, void *data)
853 appcore_ui_base_window_on_visibility(type, event);
856 static void __run(void *data)
861 static void __exit(void *data)
866 EXPORT_API appcore_ui_base_ops appcore_ui_base_get_default_ops(void)
868 appcore_ui_base_ops ops;
870 ops.base = appcore_base_get_default_ops();
872 /* override methods */
873 ops.base.create = __on_create;
874 ops.base.terminate = __on_terminate;
875 ops.base.receive = __on_receive;
876 ops.base.run = __run;
877 ops.base.exit = __exit;
879 ops.pause = __on_pause;
880 ops.resume = __on_resume;
881 ops.window.show = __window_on_show;
882 ops.window.hide = __window_on_hide;
883 ops.window.lower = __window_on_lower;
884 ops.window.visibility = __window_on_visibility;