From: gs86.lee Date: Mon, 7 Mar 2016 03:57:30 +0000 (+0900) Subject: Implement IVI profile X-Git-Tag: submit/tizen_ivi/20160308.033950^0 X-Git-Url: http://review.tizen.org/git/?a=commitdiff_plain;h=92c7853f2e18a429e979f50f949963da333db878;p=apps%2Fnative%2Fstarter.git Implement IVI profile Change-Id: I091204327b687b7344d4093dc63353b930d0c667 --- diff --git a/CMakeLists.txt b/CMakeLists.txt index 5270f4c..6e997de 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -133,6 +133,49 @@ pkg_check_modules(pkgs REQUIRED dbus-glib-1 ) +ELSEIF("${TIZEN_PROFILE_NAME}" STREQUAL "IVI") + +INCLUDE_DIRECTORIES( + ${CMAKE_SOURCE_DIR}/include + ${CMAKE_SOURCE_DIR}/include/ivi +) +INCLUDE(FindPkgConfig) +pkg_check_modules(pkgs REQUIRED + aul + capi-system-media-key + db-util + dlog + ecore + ecore-wayland + ecore-evas + ecore-input + edbus + eina + elementary + evas + syspopup-caller + vconf + appcore-efl + glib-2.0 + ui-gadget-1 + bundle + capi-appfw-application + capi-appfw-app-manager + capi-network-bluetooth + capi-system-system-settings + feedback + alarm-service + pkgmgr-info + deviced + edbus + dbus-1 + dbus-glib-1 + tts + capi-message-port + security-manager + efl-extension +) + ENDIF() IF(X11_SUPPORT) @@ -231,6 +274,22 @@ SET(BUILD_SOURCE src/wearable/starter.c ) +ELSEIF("${TIZEN_PROFILE_NAME}" STREQUAL "IVI") + +SET(BUILD_SOURCE + src/dbus_util.c + src/package_mgr.c + src/process_mgr.c + src/status.c + + src/ivi/home_mgr.c + src/ivi/lock_mgr.c + src/ivi/starter.c + src/ivi/window_mgr.c + src/ivi/popup.c + src/ivi/hw_key.c +) + ENDIF() ADD_EXECUTABLE(${PROJECT_NAME} ${BUILD_SOURCE}) diff --git a/include/ivi/home_mgr.h b/include/ivi/home_mgr.h new file mode 100644 index 0000000..6fd5061 --- /dev/null +++ b/include/ivi/home_mgr.h @@ -0,0 +1,31 @@ +/* + * Copyright (c) 2000 - 2015 Samsung Electronics Co., Ltd. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include +#include +#include + +extern int home_mgr_get_home_pid(void); +extern int home_mgr_get_volume_pid(void); + +extern void home_mgr_init(void *data); +extern void home_mgr_fini(void); + +void home_mgr_relaunch_homescreen(void); +void home_mgr_relaunch_volume(void); +extern int home_mgr_open_home(const char *pkgname); + +// End of a file diff --git a/include/ivi/hw_key.h b/include/ivi/hw_key.h new file mode 100644 index 0000000..787a779 --- /dev/null +++ b/include/ivi/hw_key.h @@ -0,0 +1,84 @@ +/* + * Copyright (c) 2000 - 2015 Samsung Electronics Co., Ltd. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#ifndef __HW_KEY_H__ +#define __HW_KEY_H__ + +#if HAVE_X11 + +#define KEY_VOLUMEUP "XF86AudioRaiseVolume" +#define KEY_VOLUMEDOWN "XF86AudioLowerVolume" +#define KEY_HOME "XF86Home" +#define KEY_CONFIG "XF86Camera_Full" +#define KEY_SEARCH "XF86Search" +#define KEY_MEDIA "XF86AudioMedia" +#define KEY_TASKSWITCH "XF86TaskPane" +#define KEY_WEBPAGE "XF86WWW" +#define KEY_MAIL "XF86Mail" +#define KEY_VOICE "XF86Voice" +#define KEY_APPS "XF86Apps" +#define KEY_CONNECT "XF86Call" +#define KEY_BACK "XF86Back" + +#elif HAVE_WAYLAND + +typedef enum { + KEY_VOLUMEUP = 0, + KEY_VOLUMEDOWN = 1, + KEY_POWER = 2, + KEY_MENU, + KEY_HOME, + KEY_BACK, + KLEY_CAMERA, + KEY_CONFIG, + KEY_SEARCH, + KEY_PLAYCD, + KEY_PAUSECB, + KEY_STOPCD, + KEY_NEXTSONG, + KEY_PRIVEIOUSSONG, + KEY_REWIND, + KEY_FASTFORWARD, + KEY_MEDIA, + KEY_PLAYPAUSE, + KEY_MUTE, + KEY_REC, + KEY_CANCEL, + KEY_SOFTBD, + KEY_QUICKPANEL, + KEY_TASKSWITCH, + KEY_HOMEPAGE, + KEY_WEBPAGE, + KEY_MAIL, + KEY_SCREENSAVER, + KEY_BRIGHTNESSDOWN, + KEY_BRIGHTNESSUP, + KEY_VOICE, + KEY_LANGUAGE, + KEY_APPS, + KEY_CONNECT, + KEY_GAMEPLAY, + KEY_VOICEWAKEUP_LPSD, + KEY_VOICEWAKEUP, + KEY_NAME_MAX, +} key_name_e; +#endif + +extern void hw_key_destroy_window(void); +extern void hw_key_create_window(void); + +#endif +// End of a file diff --git a/include/ivi/lock_mgr.h b/include/ivi/lock_mgr.h new file mode 100644 index 0000000..6db7a69 --- /dev/null +++ b/include/ivi/lock_mgr.h @@ -0,0 +1,62 @@ +/* + * Copyright (c) 2000 - 2015 Samsung Electronics Co., Ltd. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#ifndef __LOCK_DAEMON_H__ +#define __LOCK_DAEMON_H__ + +#include +#include +#include + +#include "window_mgr.h" + +#define _EDJ(x) elm_layout_edje_get(x) + +#ifdef TIZEN_BUILD_EMULATOR +#define LOCK_MGR_DEFAULT_BG_PATH "/opt/share/settings/Wallpapers/Default.jpg" +#else +#define LOCK_MGR_DEFAULT_BG_PATH "/opt/share/settings/Wallpapers/Lock_default.png" +#endif + + + +typedef enum { + LOCK_SOUND_LOCK, + LOCK_SOUND_UNLOCK, + LOCK_SOUND_BTN_KEY, + LOCK_SOUND_TAP, + LOCK_SOUND_MAX, +} lock_sound_type_e; + +typedef enum { + LCD_STATE_ON, + LCD_STATE_OFF, + LCD_STATE_MAX, +} lock_lcd_state_e; + +int lock_mgr_lcd_state_get(void); +int lock_mgr_get_lock_pid(void); +void lock_mgr_sound_play(lock_sound_type_e type); + +void lock_mgr_idle_lock_state_set(int lock_state); +Eina_Bool lock_mgr_lockscreen_launch(void); +void lock_mgr_unlock(void); + + +int lock_mgr_daemon_start(void); +void lock_mgr_daemon_end(void); + +#endif /* __LOCK_DAEMON_H__ */ diff --git a/include/ivi/popup.h b/include/ivi/popup.h new file mode 100644 index 0000000..f8602ad --- /dev/null +++ b/include/ivi/popup.h @@ -0,0 +1,25 @@ +/* + * Copyright (c) 2000 - 2015 Samsung Electronics Co., Ltd. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#ifndef __POPUP_H__ +#define __POPUP_H__ + +#include + +extern Evas_Object *popup_create(const char *title, const char *text); + +#endif +// End of a file diff --git a/include/ivi/starter.h b/include/ivi/starter.h new file mode 100644 index 0000000..a9700c2 --- /dev/null +++ b/include/ivi/starter.h @@ -0,0 +1,27 @@ +/* + * Copyright (c) 2000 - 2015 Samsung Electronics Co., Ltd. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#ifndef __STARTER_H__ +#define __STARTER_H__ + +#include + +struct appdata { + struct timeval tv_start; /* start time */ + int lcd_status; +}; + +#endif /* __STARTER_H__ */ diff --git a/packaging/starter.spec b/packaging/starter.spec index b6fac06..9bc8341 100644 --- a/packaging/starter.spec +++ b/packaging/starter.spec @@ -38,6 +38,12 @@ BuildRequires: tts-devel BuildRequires: pkgconfig(capi-message-port) BuildRequires: pkgconfig(security-manager) BuildRequires: pkgconfig(efl-extension) +%else if "%{profile}" == "ivi" +BuildRequires: tts +BuildRequires: tts-devel +BuildRequires: pkgconfig(capi-message-port) +BuildRequires: pkgconfig(security-manager) +BuildRequires: pkgconfig(efl-extension) %endif BuildRequires: pkgconfig(feedback) @@ -102,6 +108,13 @@ export CFLAGS="$CFLAGS -DTIZEN_PROFILE_WEARABLE" export CXXFLAGS="$CXXFLAGS -DTIZEN_PROFILE_WEARABLE" %endif +%if "%{profile}" == "ivi" +%define TIZEN_PROFILE_NAME "IVI" +export CFLAGS="$CFLAGS -DTIZEN_PROFILE_IVI" +export CXXFLAGS="$CXXFLAGS -DTIZEN_PROFILE_IVI" +%endif + + %ifarch %{arm} export CFLAGS="$CFLAGS -DTIZEN_BUILD_TARGET" export CXXFLAGS="$CXXFLAGS -DTIZEN_BUILD_TARGET" diff --git a/src/ivi/home_mgr.c b/src/ivi/home_mgr.c new file mode 100644 index 0000000..d6b4cd6 --- /dev/null +++ b/src/ivi/home_mgr.c @@ -0,0 +1,439 @@ +/* + * Copyright (c) 2000 - 2015 Samsung Electronics Co., Ltd. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "util.h" +#include "dbus_util.h" +#include "status.h" +#include "process_mgr.h" +#include "popup.h" + +#define VOLUME_ENABLE 0 + +#define HOME_TERMINATED "home_terminated" +#define ISTRUE "TRUE" +#define SYSPOPUPID_VOLUME "volume" + +#define DEAD_TIMER_SEC 10.0 +#define DEAD_TIMER_COUNT_MAX 2 + + + +int errno; +static struct { + pid_t home_pid; + pid_t volume_pid; + int power_off; + + Ecore_Timer *dead_timer; + int dead_count; + + Evas_Object *popup; +} s_home_mgr = { + .home_pid = (pid_t)-1, + .volume_pid = (pid_t)-1, + .power_off = 0, + + .dead_timer = NULL, + .dead_count = 0, + + .popup = NULL, +}; + + +int home_mgr_get_home_pid(void) +{ + return s_home_mgr.home_pid; +} + + + +int home_mgr_get_volume_pid(void) +{ + return s_home_mgr.volume_pid; +} + + + +static void _after_launch_home(int pid) +{ + if (dbus_util_send_oomadj(pid, OOM_ADJ_VALUE_HOMESCREEN) < 0){ + _E("failed to send oom dbus signal"); + } + s_home_mgr.home_pid = pid; +} + + + +static int _change_home_cb(const char *appid, const char *key, const char *value, void *cfn, void *afn) +{ + if (!strcmp(appid, MENU_SCREEN_PKG_NAME)) { + _E("We cannot do anything anymore."); + } else if (!strcmp(appid, status_active_get()->setappl_selected_package_name)) { + if (vconf_set_str(VCONFKEY_SETAPPL_SELECTED_PACKAGE_NAME, MENU_SCREEN_PKG_NAME) != 0) { + _E("cannot set the vconf key as %s", MENU_SCREEN_PKG_NAME); + } + /* change_home func will be called by changing the home */ + return 0; + } + _E("cannot change home"); + return -1; +} + + + +#define SERVICE_OPERATION_MAIN_KEY "__APP_SVC_OP_TYPE__" +#define SERVICE_OPERATION_MAIN_VALUE "http://tizen.org/appcontrol/operation/main" +void home_mgr_open_home(const char *appid) +{ + char *home_appid = NULL; + + if (!appid) { + home_appid = status_active_get()->setappl_selected_package_name; + } else { + home_appid = (char *) appid; + } + ret_if(!home_appid); + + process_mgr_must_launch(home_appid, SERVICE_OPERATION_MAIN_KEY, SERVICE_OPERATION_MAIN_VALUE, _change_home_cb, _after_launch_home); +} + + + +static int _show_home_cb(status_active_key_e key, void *data) +{ + int seq = status_active_get()->starter_sequence; + int is_fallback = 0; + + _D("[MENU_DAEMON] _show_home_cb is invoked(%d)", seq); + + switch (seq) { + case 0: + if (s_home_mgr.home_pid > 0) { + _D("Home[%d] has to be terminated.", s_home_mgr.home_pid); + if (aul_terminate_pid(s_home_mgr.home_pid) != AUL_R_OK) { + _E("Failed to terminate %d", s_home_mgr.home_pid); + } + s_home_mgr.home_pid = -1; /* to freeze the dead_cb */ + } + break; + case 1: + if (vconf_get_int(VCONFKEY_STARTER_IS_FALLBACK, &is_fallback) < 0) { + _E("Failed to get vconfkey : %s", VCONFKEY_STARTER_IS_FALLBACK); + } + + if (is_fallback) { + if (vconf_set_int(VCONFKEY_STARTER_IS_FALLBACK, 0)) { + _E("Failed to set vconfkey : %s", VCONFKEY_STARTER_IS_FALLBACK); + } + + if (!strcmp(status_active_get()->setappl_selected_package_name, MENU_SCREEN_PKG_NAME)) { + char *fallback_pkg; + + fallback_pkg = vconf_get_str(VCONFKEY_STARTER_FALLBACK_PKG); + _D("fallback pkg : %s", fallback_pkg); + if (fallback_pkg) { + int status; + + status = vconf_set_str(VCONFKEY_SETAPPL_SELECTED_PACKAGE_NAME, fallback_pkg); + free(fallback_pkg); + if (status == 0) { + break; + } + _E("Failed to set vconfkey : %s (%d)", VCONFKEY_SETAPPL_SELECTED_PACKAGE_NAME, status); + } else { + _E("Failed to get vconfkey : %s", VCONFKEY_STARTER_FALLBACK_PKG); + } + } + } + + home_mgr_open_home(NULL); + break; + default: + _E("False sequence [%d]", seq); + break; + } + + return 1; +} + + + +static int _change_selected_package_name(status_active_key_e key, void *data) +{ + char *appid = NULL; + int seq = status_active_get()->starter_sequence; + + if (seq < 1) { + _E("Sequence is not ready yet, do nothing"); + return 1; + } + + _D("_change_selected_package_name is invoked"); + + appid = status_active_get()->setappl_selected_package_name; + if (!appid) { + return 1; + } + _SECURE_D("pkg_name : %s", appid); + + if (s_home_mgr.home_pid > 0) { + char old_appid[BUF_SIZE_512] = { 0 , }; + + if (aul_app_get_pkgname_bypid(s_home_mgr.home_pid, old_appid, sizeof(old_appid)) == AUL_R_OK) { + if (!strcmp(appid, old_appid)) { + _D("Package is changed but same package is selected"); + return 1; + } + } + + if (AUL_R_OK != aul_terminate_pid(s_home_mgr.home_pid)) { + _D("Failed to terminate pid %d", s_home_mgr.home_pid); + } + s_home_mgr.home_pid = -1; + s_home_mgr.dead_count = 0; + if (s_home_mgr.dead_timer) { + ecore_timer_del(s_home_mgr.dead_timer); + s_home_mgr.dead_timer = NULL; + } + } + + home_mgr_open_home(appid); + + return 1; +} + + + +#if VOLUME_ENABLE +static void _after_launch_volume(int pid) +{ + if (dbus_util_send_oomadj(pid, OOM_ADJ_VALUE_DEFAULT) < 0){ + _E("failed to send oom dbus signal"); + } + s_home_mgr.volume_pid = pid; +} +#endif + + + +static void _launch_after_home(int pid) +{ + if (pid > 0) { + if(dbus_util_send_oomadj(pid, OOM_ADJ_VALUE_HOMESCREEN) < 0){ + _E("failed to send oom dbus signal"); + } + } + s_home_mgr.home_pid = pid; +} + + + +static void _launch_home(const char *appid) +{ + const char *home_appid = NULL; + + if (!appid) { + home_appid = status_active_get()->setappl_selected_package_name; + } else { + home_appid = (char *) appid; + } + ret_if(!home_appid); + + process_mgr_must_launch(home_appid, HOME_TERMINATED, ISTRUE, _change_home_cb, _launch_after_home); +} + + + +static void _popup_del_cb(void *data, Evas *e, Evas_Object *obj, void *event_info) +{ + _D("popup is deleted"); + + s_home_mgr.popup = NULL; +} + + + +static Eina_Bool _dead_timer_cb(void *data) +{ + Evas_Object *popup = NULL; + char title[BUF_SIZE_128] = { 0, }; + char text[BUF_SIZE_1024] = { 0, }; + char *appid = NULL; + + appid = status_active_get()->setappl_selected_package_name; + if (!appid) { + _E("appid is NULL"); + return ECORE_CALLBACK_CANCEL; + } + + _D("dead count : %s(%d)", appid, s_home_mgr.dead_count); + + if (s_home_mgr.dead_count >= DEAD_TIMER_COUNT_MAX) { + _D("Change homescreen package to default"); + + /* set fallback status */ + if (vconf_set_int(VCONFKEY_STARTER_IS_FALLBACK, 1) < 0) { + _E("Failed to set vconfkey : %s", VCONFKEY_STARTER_IS_FALLBACK); + } + + if (vconf_set_str(VCONFKEY_STARTER_FALLBACK_PKG, appid) < 0) { + _E("Failed to set vconfkey : %s", VCONFKEY_STARTER_FALLBACK_PKG); + } + + strncpy(title, _("IDS_COM_POP_WARNING"), sizeof(title)); + title[sizeof(title) - 1] = '\0'; + + snprintf(text, sizeof(text), _("IDS_IDLE_POP_UNABLE_TO_LAUNCH_PS"), appid); + _D("title : %s / text : %s", title, text); + + if (!s_home_mgr.popup) { + popup = popup_create(title, text); + if (!popup) { + _E("Failed to create popup"); + } else { + s_home_mgr.popup = popup; + evas_object_event_callback_add(popup, EVAS_CALLBACK_DEL, _popup_del_cb, NULL); + } + } + + if (vconf_set_str(VCONFKEY_SETAPPL_SELECTED_PACKAGE_NAME, MENU_SCREEN_PKG_NAME) != 0) { + _E("cannot set the vconf key as %s", MENU_SCREEN_PKG_NAME); + return ECORE_CALLBACK_RENEW; + } + } + + s_home_mgr.dead_timer = NULL; + s_home_mgr.dead_count = 0; + + return ECORE_CALLBACK_CANCEL; +} + + + +void home_mgr_relaunch_homescreen(void) +{ + char *appid = NULL; + + if (s_home_mgr.power_off) { + _E("power off"); + return; + } + + appid = status_active_get()->setappl_selected_package_name; + if (!appid) { + _E("appid is NULL"); + return; + } + + s_home_mgr.dead_count++; + _D("home dead count : %d", s_home_mgr.dead_count); + + if (!s_home_mgr.dead_timer) { + _D("Add dead timer"); + s_home_mgr.dead_timer = ecore_timer_add(DEAD_TIMER_SEC, _dead_timer_cb, NULL); + if (!s_home_mgr.dead_timer) { + _E("Failed to add a dead timer"); + } + } + + _launch_home(appid); +} + + + +void home_mgr_relaunch_volume(void) +{ +#if VOLUME_ENABLE + process_mgr_must_syspopup_launch(SYSPOPUPID_VOLUME, NULL, NULL, NULL, _after_launch_volume); +#endif +} + + + +static int _power_off_cb(status_active_key_e key, void *data) +{ + int val = status_active_get()->sysman_power_off_status; + + if (val == VCONFKEY_SYSMAN_POWER_OFF_DIRECT || val == VCONFKEY_SYSMAN_POWER_OFF_RESTART) { + s_home_mgr.power_off = 1; + } else { + s_home_mgr.power_off = 0; + } + + _D("power off status : %d", s_home_mgr.power_off); + + return 1; +} + + + +#if VOLUME_ENABLE +static Eina_Bool _launch_volume_idler_cb(void *data) +{ + process_mgr_must_syspopup_launch(SYSPOPUPID_VOLUME, NULL, NULL, NULL, _after_launch_volume); + return ECORE_CALLBACK_CANCEL; +} +#endif + + + +void home_mgr_init(void *data) +{ + _D( "[MENU_DAEMON]home_mgr_init is invoked"); + + status_active_register_cb(STATUS_ACTIVE_KEY_STARTER_SEQUENCE, _show_home_cb, NULL); + status_active_register_cb(STATUS_ACTIVE_KEY_SYSMAN_POWER_OFF_STATUS, _power_off_cb, NULL); + status_active_register_cb(STATUS_ACTIVE_KEY_SETAPPL_SELECTED_PACKAGE_NAME, _change_selected_package_name, NULL); + _change_selected_package_name(STATUS_ACTIVE_KEY_SETAPPL_SELECTED_PACKAGE_NAME, NULL); + +#if VOLUME_ENABLE + ecore_idler_add(_launch_volume_idler_cb, NULL); +#endif +} + + + +void home_mgr_fini(void) +{ +#if VOLUME_ENABLE + if (s_home_mgr.volume_pid > 0) { + process_mgr_terminate_app(s_home_mgr.volume_pid, 1); + s_home_mgr.volume_pid = -1; + } +#endif + + status_active_unregister_cb(STATUS_ACTIVE_KEY_STARTER_SEQUENCE, _show_home_cb); + status_active_unregister_cb(STATUS_ACTIVE_KEY_SYSMAN_POWER_OFF_STATUS, _power_off_cb); + status_active_unregister_cb(STATUS_ACTIVE_KEY_SETAPPL_SELECTED_PACKAGE_NAME, _change_selected_package_name); +} + + + +// End of a file diff --git a/src/ivi/hw_key.c b/src/ivi/hw_key.c new file mode 100644 index 0000000..446b4d3 --- /dev/null +++ b/src/ivi/hw_key.c @@ -0,0 +1,959 @@ +/* + * Copyright (c) 2000 - 2015 Samsung Electronics Co., Ltd. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#ifdef HAVE_X11 + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "hw_key.h" +#include "home_mgr.h" +#include "util.h" +#include "dbus_util.h" +#include "lock_mgr.h" +#include "status.h" +#include "process_mgr.h" +#include "lock_pwd_util.h" + +#define APPID_CAMERA "org.tizen.camera-app" +#define APPID_CALLLOG "org.tizen.calllog" +#define APPID_MUSIC_PLAYER "org.tizen.music-player" +#define APPID_TASKMGR "org.tizen.task-mgr" +#define APPID_BROWSER "org.tizen.browser" +#define APPID_EMAIL "org.tizen.email" +#define APPID_DIALER "org.tizen.phone" + +#define STR_ATOM_XKEY_COMPOSITION "_XKEY_COMPOSITION" +#define STR_ATOM_KEYROUTER_NOTIWINDOW "_KEYROUTER_NOTIWINDOW" + +#define LONG_PRESS_TIMER_SEC 0.4 +#define HOMEKEY_TIMER_SEC 0.2 +#define CANCEL_KEY_TIMER_SEC 0.3 + +static struct { + Ecore_X_Window win; + Ecore_Event_Handler *key_up; + Ecore_Event_Handler *key_down; + Ecore_Timer *home_long_press_timer; + Ecore_Timer *home_multi_press_timer; + Eina_Bool cancel; + Ecore_X_Window keyrouter_notiwindow; + int homekey_count; +} key_info = { + .win = 0x0, + .key_up = NULL, + .key_down = NULL, + .home_long_press_timer = NULL, + .home_multi_press_timer = NULL, + .cancel = EINA_FALSE, + .keyrouter_notiwindow = 0x0, + .homekey_count = 0, +}; + + + +static void _after_launch_taskmgr(int pid) +{ + if(0 < pid) { + if(dbus_util_send_oomadj(pid, OOM_ADJ_VALUE_DEFAULT) < 0){ + _E("failed to send oom dbus signal"); + } + } +} + + + +static Eina_Bool _launch_taskmgr_cb(void* data) +{ + int val = -1; + + _D("Launch TASKMGR"); + + key_info.home_long_press_timer = NULL; + + if (vconf_get_int(VCONFKEY_IDLE_LOCK_STATE, &val) < 0) { + _E("Cannot get VCONFKEY for lock state"); + } else if (VCONFKEY_IDLE_LOCK == val) { + _E("lock state, ignore home key long press..!!"); + return ECORE_CALLBACK_CANCEL; + } + + process_mgr_must_launch(APPID_TASKMGR, NULL, NULL, NULL, _after_launch_taskmgr); + + return ECORE_CALLBACK_CANCEL; +} + + + +static void _release_multimedia_key(const char *value) +{ + ret_if(NULL == value); + _D("Multimedia key is released with %s", value); + process_mgr_must_launch(APPID_MUSIC_PLAYER, "multimedia_key", value, NULL, NULL); +} + + + +static Eina_Bool _launch_by_home_key(void *data) +{ + int ret = 0; + + if (status_passive_get()->idle_lock_state > VCONFKEY_IDLE_UNLOCK) { + return ECORE_CALLBACK_CANCEL; + } + + ret = home_mgr_open_home(NULL); + if(ret > 0) { + dbus_util_send_home_raise_signal(); + } + + return ECORE_CALLBACK_CANCEL; +} + + + +static Eina_Bool _home_multi_press_timer_cb(void *data) +{ + _W("homekey count[%d]", key_info.homekey_count); + + key_info.home_multi_press_timer = NULL; + + if(0 == key_info.homekey_count % 2) { + key_info.homekey_count = 0; + return ECORE_CALLBACK_CANCEL; + } else if(key_info.homekey_count >= 3) { + key_info.homekey_count = 0; + return ECORE_CALLBACK_CANCEL; + } + + /* Single homekey operation */ + key_info.homekey_count = 0; + _launch_by_home_key(data); + + return ECORE_CALLBACK_CANCEL; + +} + + + +#define SERVICE_OPERATION_POPUP_SEARCH "http://samsung.com/appcontrol/operation/search" +#define SEARCH_PKG_NAME "org.tizen.sfinder" +static int _launch_search(void) +{ + app_control_h app_control; + int ret = APP_CONTROL_ERROR_NONE; + + app_control_create(&app_control); + app_control_set_operation(app_control, APP_CONTROL_OPERATION_DEFAULT); + app_control_set_app_id(app_control, SEARCH_PKG_NAME); + + ret = app_control_send_launch_request(app_control, NULL, NULL); + + if(ret != APP_CONTROL_ERROR_NONE) { + _E("Cannot launch search!! err[%d]", ret); + } + + app_control_destroy(app_control); + return ret; +} + + + +static void _cancel_key_events(void) +{ + key_info.homekey_count = 0; + + if (key_info.home_long_press_timer) { + ecore_timer_del(key_info.home_long_press_timer); + key_info.home_long_press_timer = NULL; + } + + if(key_info.home_multi_press_timer) { + ecore_timer_del(key_info.home_multi_press_timer); + key_info.home_multi_press_timer = NULL; + } +} + + + +static Eina_Bool _key_release_cb(void *data, int type, void *event) +{ + Evas_Event_Key_Up *ev = event; + + retv_if(!ev, ECORE_CALLBACK_RENEW); + retv_if(!ev->keyname, ECORE_CALLBACK_RENEW); + + _D("_key_release_cb : %s Released", ev->keyname); + + /* Priority 1 : Cancel event */ + if (!strcmp(ev->keyname, KEY_CANCEL)) { + _D("CANCEL Key is released"); + key_info.cancel = EINA_FALSE; + return ECORE_CALLBACK_RENEW; + } + + if (EINA_TRUE == key_info.cancel) { + _D("CANCEL is on"); + return ECORE_CALLBACK_RENEW; + } + + /* Priority 2 : Execute before checking the lock status */ + if (!strcmp(ev->keyname, KEY_MEDIA)) { + _release_multimedia_key("KEY_PLAYCD"); + return ECORE_CALLBACK_RENEW; + } + + /* Priority 3 : Check the lock status */ + if ((status_passive_get()->idle_lock_state == VCONFKEY_IDLE_LOCK) + && (status_active_get()->setappl_screen_lock_type_int > SETTING_SCREEN_LOCK_TYPE_NONE)) { + if (!strcmp(ev->keyname, KEY_BACK)) { + _D("Back key is released"); + lock_pwd_util_back_key_relased(); + } else { + _D("phone lock state, ignore home key."); + } + return ECORE_CALLBACK_RENEW; + } + + /* Priority 4 : These keys are only activated after checking the lock state */ + if (!strcmp(ev->keyname, KEY_END)) { + } else if (!strcmp(ev->keyname, KEY_CONFIG)) { + } else if (!strcmp(ev->keyname, KEY_SEND)) { + } else if (!strcmp(ev->keyname, KEY_HOME)) { + _W("Home Key is released"); + + syspopup_destroy_all(); + + if(key_info.home_multi_press_timer) { + _D("delete homekey timer"); + ecore_timer_del(key_info.home_multi_press_timer); + key_info.home_multi_press_timer = NULL; + } + + if (key_info.home_long_press_timer) { + ecore_timer_del(key_info.home_long_press_timer); + key_info.home_long_press_timer = NULL; + } else { + key_info.homekey_count = 0; + return ECORE_CALLBACK_RENEW; + } + + key_info.home_multi_press_timer = ecore_timer_add(HOMEKEY_TIMER_SEC, _home_multi_press_timer_cb, NULL); + if (!key_info.home_multi_press_timer) { + _E("Critical! cannot add a timer for home multi press"); + } + return ECORE_CALLBACK_RENEW; + } else if (!strcmp(ev->keyname, KEY_PAUSE)) { + } else if (!strcmp(ev->keyname, KEY_APPS)) { + _D("App tray key is released"); + } else if (!strcmp(ev->keyname, KEY_TASKSWITCH)) { + _D("Task switch key is released"); + _launch_taskmgr_cb(NULL); + } else if (!strcmp(ev->keyname, KEY_WEBPAGE)) { + _D("Web page key is released"); + process_mgr_must_open(APPID_BROWSER, NULL, NULL); + } else if (!strcmp(ev->keyname, KEY_MAIL)) { + _D("Mail key is released"); + process_mgr_must_open(APPID_EMAIL, NULL, NULL); + } else if (!strcmp(ev->keyname, KEY_CONNECT)) { + _D("Connect key is released"); + process_mgr_must_open(APPID_DIALER, NULL, NULL); + } else if (!strcmp(ev->keyname, KEY_SEARCH)) { + _D("Search key is released"); + if (_launch_search() < 0) { + _E("Failed to launch the search"); + } + } else if (!strcmp(ev->keyname, KEY_VOICE)) { + _D("Voice key is released"); + } + + return ECORE_CALLBACK_RENEW; +} + + + +static Eina_Bool _key_press_cb(void *data, int type, void *event) +{ + Evas_Event_Key_Down *ev = event; + + retv_if(!ev, ECORE_CALLBACK_RENEW); + retv_if(!ev->keyname, ECORE_CALLBACK_RENEW); + + _D("_key_press_cb : %s Pressed", ev->keyname); + + /* Priority 1 : Cancel */ + /* every reserved events have to be canceld when cancel key is pressed */ + if (!strcmp(ev->keyname, KEY_CANCEL)) { + _D("Cancel button is pressed"); + key_info.cancel = EINA_TRUE; + _cancel_key_events(); + return ECORE_CALLBACK_RENEW; + } + + if (EINA_TRUE == key_info.cancel) { + _D("CANCEL is on"); + return ECORE_CALLBACK_RENEW; + } + + /* Priority 2 : Check the lock status */ + if ((status_passive_get()->idle_lock_state == VCONFKEY_IDLE_LOCK) + && (status_active_get()->setappl_screen_lock_type_int > SETTING_SCREEN_LOCK_TYPE_NONE)) { + _D("phone lock state, ignore key events."); + _cancel_key_events(); + return ECORE_CALLBACK_RENEW; + } + + /* Priority 3 : other keys */ + if (!strcmp(ev->keyname, KEY_SEND)) { + _D("Launch calllog"); + process_mgr_must_open(APPID_CALLLOG, NULL, NULL); + } else if(!strcmp(ev->keyname, KEY_CONFIG)) { + _D("Launch camera"); + process_mgr_must_open(APPID_CAMERA, NULL, NULL); + } else if (!strcmp(ev->keyname, KEY_HOME)) { + _W("Home Key is pressed"); + if (key_info.home_long_press_timer) { + ecore_timer_del(key_info.home_long_press_timer); + key_info.home_long_press_timer = NULL; + } + + key_info.homekey_count++; + _W("homekey count : %d", key_info.homekey_count); + + if(key_info.home_multi_press_timer) { + ecore_timer_del(key_info.home_multi_press_timer); + key_info.home_multi_press_timer = NULL; + _D("delete homekey timer"); + } + + _D("create long press timer"); + key_info.home_long_press_timer = ecore_timer_add(LONG_PRESS_TIMER_SEC, _launch_taskmgr_cb, NULL); + if (!key_info.home_long_press_timer) { + _E("Failed to add timer for long press detection"); + } + } else if (!strcmp(ev->keyname, KEY_MEDIA)) { + _D("Media key is pressed"); + } else if (!strcmp(ev->keyname, KEY_APPS)) { + _D("App tray key is pressed"); + } else if (!strcmp(ev->keyname, KEY_TASKSWITCH)) { + _D("Task switch key is pressed"); + } else if (!strcmp(ev->keyname, KEY_WEBPAGE)) { + _D("Web page key is pressed"); + } else if (!strcmp(ev->keyname, KEY_MAIL)) { + _D("Mail key is pressed"); + } else if (!strcmp(ev->keyname, KEY_SEARCH)) { + _D("Search key is pressed"); + } else if (!strcmp(ev->keyname, KEY_VOICE)) { + _D("Voice key is pressed"); + } else if (!strcmp(ev->keyname, KEY_CONNECT)) { + _D("Connect key is pressed"); + } + + return ECORE_CALLBACK_RENEW; +} + + + +void _media_key_event_cb(media_key_e key, media_key_event_e status, void *user_data) +{ + _D("MEDIA KEY EVENT : %d", key); + if (MEDIA_KEY_STATUS_PRESSED == status) return; + + switch (key) { + case MEDIA_KEY_PAUSE: + _release_multimedia_key("KEY_PAUSECD"); + break; + case MEDIA_KEY_PLAY: + _release_multimedia_key("KEY_PLAYCD"); + break; + case MEDIA_KEY_PLAYPAUSE: + _release_multimedia_key("KEY_PLAYPAUSECD"); + break; + default: + _E("cannot reach here, key[%d]", key); + break; + } +} + + + +void hw_key_create_window(void) +{ + int ret; + Ecore_X_Atom atomNotiWindow; + Ecore_X_Window keyrouter_notiwindow; + + key_info.win = ecore_x_window_input_new(0, 0, 0, 1, 1); + if (!key_info.win) { + _D("Failed to create hidden window"); + return; + } + ecore_x_event_mask_unset(key_info.win, ECORE_X_EVENT_MASK_NONE); + ecore_x_icccm_title_set(key_info.win, "menudaemon,key,receiver"); + ecore_x_netwm_name_set(key_info.win, "menudaemon,key,receiver"); + ecore_x_netwm_pid_set(key_info.win, getpid()); + + utilx_grab_key(ecore_x_display_get(), key_info.win, KEY_HOME, SHARED_GRAB); + utilx_grab_key(ecore_x_display_get(), key_info.win, KEY_VOLUMEDOWN, SHARED_GRAB); + utilx_grab_key(ecore_x_display_get(), key_info.win, KEY_VOLUMEUP, SHARED_GRAB); + utilx_grab_key(ecore_x_display_get(), key_info.win, KEY_CONFIG, SHARED_GRAB); + utilx_grab_key(ecore_x_display_get(), key_info.win, KEY_MEDIA, SHARED_GRAB); + utilx_grab_key(ecore_x_display_get(), key_info.win, KEY_APPS, SHARED_GRAB); + utilx_grab_key(ecore_x_display_get(), key_info.win, KEY_TASKSWITCH, SHARED_GRAB); + utilx_grab_key(ecore_x_display_get(), key_info.win, KEY_WEBPAGE, SHARED_GRAB); + utilx_grab_key(ecore_x_display_get(), key_info.win, KEY_MAIL, SHARED_GRAB); + utilx_grab_key(ecore_x_display_get(), key_info.win, KEY_SEARCH, SHARED_GRAB); + utilx_grab_key(ecore_x_display_get(), key_info.win, KEY_VOICE, SHARED_GRAB); + utilx_grab_key(ecore_x_display_get(), key_info.win, KEY_CONNECT, SHARED_GRAB); + utilx_grab_key(ecore_x_display_get(), key_info.win, KEY_POWER, SHARED_GRAB); + + key_info.key_up = ecore_event_handler_add(ECORE_EVENT_KEY_UP, _key_release_cb, NULL); + if (!key_info.key_up) + _E("Failed to register a key up event handler"); + + key_info.key_down = ecore_event_handler_add(ECORE_EVENT_KEY_DOWN, _key_press_cb, NULL); + if (!key_info.key_down) + _E("Failed to register a key down event handler"); + + /* Get notifwindow */ + atomNotiWindow = ecore_x_atom_get(STR_ATOM_KEYROUTER_NOTIWINDOW); + ret = ecore_x_window_prop_window_get(ecore_x_window_root_first_get(), atomNotiWindow, &keyrouter_notiwindow, 1); + if (ret > 0) { + _D("Succeed to get keyrouter notiwindow ! ret = %d (win=0x%x)\n" + , ret, keyrouter_notiwindow); + ecore_x_window_sniff(keyrouter_notiwindow); + key_info.keyrouter_notiwindow = keyrouter_notiwindow; + } else { + _E("Failed to get keyrouter notiwindow! ret = %d, atomNotiWindow = 0x%x, keyrouter_notiwindow = 0x%x" + , ret, atomNotiWindow, keyrouter_notiwindow); + } + + media_key_reserve(_media_key_event_cb, NULL); +} + + + +void hw_key_destroy_window(void) +{ + utilx_ungrab_key(ecore_x_display_get(), key_info.win, KEY_HOME); + utilx_ungrab_key(ecore_x_display_get(), key_info.win, KEY_VOLUMEDOWN); + utilx_ungrab_key(ecore_x_display_get(), key_info.win, KEY_VOLUMEUP); + utilx_ungrab_key(ecore_x_display_get(), key_info.win, KEY_CONFIG); + utilx_ungrab_key(ecore_x_display_get(), key_info.win, KEY_MEDIA); + utilx_ungrab_key(ecore_x_display_get(), key_info.win, KEY_APPS); + utilx_ungrab_key(ecore_x_display_get(), key_info.win, KEY_TASKSWITCH); + utilx_ungrab_key(ecore_x_display_get(), key_info.win, KEY_WEBPAGE); + utilx_ungrab_key(ecore_x_display_get(), key_info.win, KEY_MAIL); + utilx_ungrab_key(ecore_x_display_get(), key_info.win, KEY_SEARCH); + utilx_ungrab_key(ecore_x_display_get(), key_info.win, KEY_VOICE); + utilx_ungrab_key(ecore_x_display_get(), key_info.win, KEY_CONNECT); + + if (key_info.key_up) { + ecore_event_handler_del(key_info.key_up); + key_info.key_up = NULL; + } + + if (key_info.key_down) { + ecore_event_handler_del(key_info.key_down); + key_info.key_down = NULL; + } + + ecore_x_window_delete_request_send(key_info.win); + key_info.win = 0x0; + + media_key_release(); +} + +#elif HAVE_WAYLAND + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "hw_key.h" +#include "home_mgr.h" +#include "util.h" +#include "dbus_util.h" +#include "lock_mgr.h" +#include "status.h" +#include "process_mgr.h" + +#define APPID_CAMERA "org.tizen.camera-app" +#define APPID_CALLLOG "org.tizen.calllog" +#define APPID_MUSIC_PLAYER "org.tizen.music-player" +#define APPID_TASKMGR "org.tizen.task-mgr" +#define APPID_BROWSER "org.tizen.browser" +#define APPID_EMAIL "org.tizen.email" +#define APPID_DIALER "org.tizen.phone" + +#define STR_ATOM_KEYROUTER_NOTIWINDOW "_KEYROUTER_NOTIWINDOW" + +#define LONG_PRESS_TIMER_SEC 0.4 +#define HOMEKEY_TIMER_SEC 0.2 +#define CANCEL_KEY_TIMER_SEC 0.3 + + +const char *key_name[38] = { + "XF86AudioRaiseVolume", + "XF86AudioLowerVolume", + "XF86PowerOff", + "XF86Menu", + "XF86Home", + "XF86Back", + "XF86Camera", + "XF86Camera_Full", + "XF86Search", + "XF86AudioPlay", + "XF86AudioPause", + "XF86AudioStop", + "XF86AudioNext", + "XF86AudioPrev", + "XF86AudioRewind", + "XF86AudioForward", + "XF86AudioMedia", + "XF86AudioPlayPause", + "XF86AudioMute", + "XF86AudioRecord", + "Cancel", + "XF86SoftKBD", + "XF86QuickPanel", + "XF86TaskPane", + "XF86HomePage", + "XF86WWW", + "XF86Mail", + "XF86ScreenSaver", + "XF86MonBrightnessDown", + "XF86MonBrightnessUp", + "XF86Voice", + "Hangul", + "XF86Apps", + "XF86Call", + "XF86Game", + "XF86VoiceWakeUp_LPSD", + "XF86VoiceWakeUp", + "KEY_NAME_MAX", +}; + + +static struct { + Ecore_Event_Handler *key_up; + Ecore_Event_Handler *key_down; + Ecore_Timer *home_long_press_timer; + Ecore_Timer *home_multi_press_timer; + Ecore_Timer *keygrab_timer; + Eina_Bool cancel; + int homekey_count; +} key_info = { + .key_up = NULL, + .key_down = NULL, + .home_long_press_timer = NULL, + .home_multi_press_timer = NULL, + .keygrab_timer = NULL, + .cancel = EINA_FALSE, + .homekey_count = 0, +}; + + + + +static void _cancel_key_events(void) +{ + key_info.homekey_count = 0; + + if (key_info.home_long_press_timer) { + ecore_timer_del(key_info.home_long_press_timer); + key_info.home_long_press_timer = NULL; + } + + if(key_info.home_multi_press_timer) { + ecore_timer_del(key_info.home_multi_press_timer); + key_info.home_multi_press_timer = NULL; + } +} + + + +#define SERVICE_OPERATION_POPUP_SEARCH "http://samsung.com/appcontrol/operation/search" +#define SEARCH_PKG_NAME "org.tizen.sfinder" +static int _launch_search(void) +{ + app_control_h app_control; + int ret = APP_CONTROL_ERROR_NONE; + + app_control_create(&app_control); + app_control_set_operation(app_control, APP_CONTROL_OPERATION_DEFAULT); + app_control_set_app_id(app_control, SEARCH_PKG_NAME); + + ret = app_control_send_launch_request(app_control, NULL, NULL); + + if(ret != APP_CONTROL_ERROR_NONE) { + _E("Cannot launch search!! err[%d]", ret); + } + + app_control_destroy(app_control); + return ret; +} + + + +static void _after_launch_taskmgr(int pid) +{ + if(0 < pid) { + if(dbus_util_send_oomadj(pid, OOM_ADJ_VALUE_DEFAULT) < 0){ + _E("failed to send oom dbus signal"); + } + } +} + + + +static Eina_Bool _launch_taskmgr_cb(void* data) +{ + int val = -1; + + _D("Launch TASKMGR"); + + key_info.home_long_press_timer = NULL; + + if (vconf_get_int(VCONFKEY_IDLE_LOCK_STATE, &val) < 0) { + _E("Cannot get VCONFKEY for lock state"); + } else if (VCONFKEY_IDLE_LOCK == val) { + _E("lock state, ignore home key long press..!!"); + return ECORE_CALLBACK_CANCEL; + } + + process_mgr_must_launch(APPID_TASKMGR, NULL, NULL, NULL, _after_launch_taskmgr); + + return ECORE_CALLBACK_CANCEL; +} + + + +static Eina_Bool _launch_by_home_key(void *data) +{ + int ret = 0; + + if (status_passive_get()->idle_lock_state > VCONFKEY_IDLE_UNLOCK) { + return ECORE_CALLBACK_CANCEL; + } + + ret = home_mgr_open_home(NULL); + if(ret > 0) { + dbus_util_send_home_raise_signal(); + } + + return ECORE_CALLBACK_CANCEL; +} + + + +static Eina_Bool _home_multi_press_timer_cb(void *data) +{ + _W("homekey count[%d]", key_info.homekey_count); + + key_info.home_multi_press_timer = NULL; + + if(0 == key_info.homekey_count % 2) { + key_info.homekey_count = 0; + return ECORE_CALLBACK_CANCEL; + } else if(key_info.homekey_count >= 3) { + key_info.homekey_count = 0; + return ECORE_CALLBACK_CANCEL; + } + + /* Single homekey operation */ + key_info.homekey_count = 0; + _launch_by_home_key(data); + + return ECORE_CALLBACK_CANCEL; + +} + + + +static void _release_multimedia_key(const char *value) +{ + ret_if(NULL == value); + _D("Multimedia key is released with %s", value); + process_mgr_must_launch(APPID_MUSIC_PLAYER, "multimedia_key", value, NULL, NULL); +} + + + +static Eina_Bool _key_release_cb(void *data, int type, void *event) +{ + Evas_Event_Key_Up *ev = event; + + retv_if(!ev, ECORE_CALLBACK_RENEW); + retv_if(!ev->keyname, ECORE_CALLBACK_RENEW); + + _D("_key_release_cb : %s Released", ev->keyname); + + /* Priority 1 : Cancel event */ + if (!strcmp(ev->keyname, key_name[KEY_CANCEL])) { + _D("CANCEL Key is released"); + key_info.cancel = EINA_FALSE; + return ECORE_CALLBACK_RENEW; + } + + if (EINA_TRUE == key_info.cancel) { + _D("CANCEL is on"); + return ECORE_CALLBACK_RENEW; + } + + /* Priority 2 : Execute before checking the lock status */ + if (!strcmp(ev->keyname, key_name[KEY_MEDIA])) { + _release_multimedia_key("KEY_PLAYCD"); + return ECORE_CALLBACK_RENEW; + } + + /* Priority 3 : Check the lock status */ + if ((status_passive_get()->idle_lock_state == VCONFKEY_IDLE_LOCK) + && (status_active_get()->setappl_screen_lock_type_int > SETTING_SCREEN_LOCK_TYPE_NONE)) { + if (!strcmp(ev->keyname, key_name[KEY_BACK])) { + _D("Back key is released"); + } else { + _D("phone lock state, ignore home key."); + } + return ECORE_CALLBACK_RENEW; + } + + /* Priority 4 : These keys are only activated after checking the lock state */ + if (!strcmp(ev->keyname, key_name[KEY_HOME])) { + _W("Home Key is released"); + + syspopup_destroy_all(); + + if(key_info.home_multi_press_timer) { + _D("delete homekey timer"); + ecore_timer_del(key_info.home_multi_press_timer); + key_info.home_multi_press_timer = NULL; + } + + if (key_info.home_long_press_timer) { + ecore_timer_del(key_info.home_long_press_timer); + key_info.home_long_press_timer = NULL; + } else { + key_info.homekey_count = 0; + return ECORE_CALLBACK_RENEW; + } + + key_info.home_multi_press_timer = ecore_timer_add(HOMEKEY_TIMER_SEC, _home_multi_press_timer_cb, NULL); + if (!key_info.home_multi_press_timer) { + _E("Critical! cannot add a timer for home multi press"); + } + return ECORE_CALLBACK_RENEW; + } else if (!strcmp(ev->keyname, key_name[KEY_APPS])) { + _D("App tray key is released"); + } else if (!strcmp(ev->keyname, key_name[KEY_TASKSWITCH])) { + _D("Task switch key is released"); + _launch_taskmgr_cb(NULL); + } else if (!strcmp(ev->keyname, key_name[KEY_WEBPAGE])) { + _D("Web page key is released"); + process_mgr_must_open(APPID_BROWSER, NULL, NULL); + } else if (!strcmp(ev->keyname, key_name[KEY_MAIL])) { + _D("Mail key is released"); + process_mgr_must_open(APPID_EMAIL, NULL, NULL); + } else if (!strcmp(ev->keyname, key_name[KEY_CONNECT])) { + _D("Connect key is released"); + process_mgr_must_open(APPID_DIALER, NULL, NULL); + } else if (!strcmp(ev->keyname, key_name[KEY_SEARCH])) { + _D("Search key is released"); + if (_launch_search() < 0) { + _E("Failed to launch the search"); + } + } else if (!strcmp(ev->keyname, key_name[KEY_VOICE])) { + _D("Voice key is released"); + } + + return ECORE_CALLBACK_RENEW; +} + + + +static Eina_Bool _key_press_cb(void *data, int type, void *event) +{ + Evas_Event_Key_Down *ev = event; + + retv_if(!ev, ECORE_CALLBACK_RENEW); + retv_if(!ev->keyname, ECORE_CALLBACK_RENEW); + + _D("_key_press_cb : %s Pressed", ev->keyname); + + /* Priority 1 : Cancel */ + /* every reserved events have to be canceld when cancel key is pressed */ + if (!strcmp(ev->keyname, key_name[KEY_CANCEL])) { + _D("Cancel button is pressed"); + key_info.cancel = EINA_TRUE; + _cancel_key_events(); + return ECORE_CALLBACK_RENEW; + } + + if (EINA_TRUE == key_info.cancel) { + _D("CANCEL is on"); + return ECORE_CALLBACK_RENEW; + } + + /* Priority 2 : Check the lock status */ + if ((status_passive_get()->idle_lock_state == VCONFKEY_IDLE_LOCK) + && (status_active_get()->setappl_screen_lock_type_int > SETTING_SCREEN_LOCK_TYPE_NONE)) { + _D("phone lock state, ignore key events."); + _cancel_key_events(); + return ECORE_CALLBACK_RENEW; + } + + /* Priority 3 : other keys */ +#if 0 + if (!strcmp(ev->keyname, key_name[KEY_SEND])) { + _D("Launch calllog"); + process_mgr_must_open(APPID_CALLLOG, NULL, NULL); + } else +#endif + if(!strcmp(ev->keyname, key_name[KEY_CONFIG])) { + _D("Launch camera"); + process_mgr_must_open(APPID_CAMERA, NULL, NULL); + } else if (!strcmp(ev->keyname, key_name[KEY_HOME])) { + _W("Home Key is pressed"); + if (key_info.home_long_press_timer) { + ecore_timer_del(key_info.home_long_press_timer); + key_info.home_long_press_timer = NULL; + } + + key_info.homekey_count++; + _W("homekey count : %d", key_info.homekey_count); + + if(key_info.home_multi_press_timer) { + ecore_timer_del(key_info.home_multi_press_timer); + key_info.home_multi_press_timer = NULL; + _D("delete homekey timer"); + } + + _D("create long press timer"); + key_info.home_long_press_timer = ecore_timer_add(LONG_PRESS_TIMER_SEC, _launch_taskmgr_cb, NULL); + if (!key_info.home_long_press_timer) { + _E("Failed to add timer for long press detection"); + } + } else if (!strcmp(ev->keyname, key_name[KEY_MEDIA])) { + _D("Media key is pressed"); + } else if (!strcmp(ev->keyname, key_name[KEY_APPS])) { + _D("App tray key is pressed"); + } else if (!strcmp(ev->keyname, key_name[KEY_TASKSWITCH])) { + _D("Task switch key is pressed"); + } else if (!strcmp(ev->keyname, key_name[KEY_WEBPAGE])) { + _D("Web page key is pressed"); + } else if (!strcmp(ev->keyname, key_name[KEY_MAIL])) { + _D("Mail key is pressed"); + } else if (!strcmp(ev->keyname, key_name[KEY_SEARCH])) { + _D("Search key is pressed"); + } else if (!strcmp(ev->keyname, key_name[KEY_VOICE])) { + _D("Voice key is pressed"); + } else if (!strcmp(ev->keyname, key_name[KEY_CONNECT])) { + _D("Connect key is pressed"); + } + + return ECORE_CALLBACK_RENEW; +} + + + +static Eina_Bool __keygrab_timer_cb(void *data) +{ + int i = 0; + int ret = 0; + + for (i = 0; i < KEY_NAME_MAX; i++) { + ret = ecore_wl_window_keygrab_set(NULL, key_name[i], 0, 0, 0, ECORE_WL_WINDOW_KEYGRAB_SHARED); + _D("key grab : %s / ret : %d", key_name[i], ret); + } + + key_info.key_up = ecore_event_handler_add(ECORE_EVENT_KEY_UP, _key_release_cb, NULL); + if (!key_info.key_up) { + _E("Failed to register a key up event handler"); + } + + key_info.key_down = ecore_event_handler_add(ECORE_EVENT_KEY_DOWN, _key_press_cb, NULL); + if (!key_info.key_down) { + _E("Failed to register a key down event handler"); + } + + return ECORE_CALLBACK_CANCEL; +} + + + +void hw_key_create_window(void) +{ + if (key_info.keygrab_timer) { + ecore_timer_del(key_info.keygrab_timer); + key_info.keygrab_timer = NULL; + } + + key_info.keygrab_timer = ecore_timer_add(1.0f, __keygrab_timer_cb, NULL); + if (!key_info.keygrab_timer) { + _E("Failed to add timer for keygrab"); + } +} + + + +void hw_key_destroy_window(void) +{ + int i = 0; + + for (i = 0; i < KEY_NAME_MAX; i++) { + ecore_wl_window_keygrab_unset(NULL, key_name[i], 0, 0); + } + + if (key_info.keygrab_timer) { + ecore_timer_del(key_info.keygrab_timer); + key_info.keygrab_timer = NULL; + } + + if (key_info.key_up) { + ecore_event_handler_del(key_info.key_up); + key_info.key_up = NULL; + } + + if (key_info.key_down) { + ecore_event_handler_del(key_info.key_down); + key_info.key_down = NULL; + } +} + +#endif + +// End of a file diff --git a/src/ivi/lock_mgr.c b/src/ivi/lock_mgr.c new file mode 100644 index 0000000..7bc8d53 --- /dev/null +++ b/src/ivi/lock_mgr.c @@ -0,0 +1,480 @@ +/* + * Copyright (c) 2000 - 2015 Samsung Electronics Co., Ltd. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "lock_mgr.h" +#include "package_mgr.h" +#include "process_mgr.h" +#include "hw_key.h" +#include "dbus_util.h" +#include "util.h" +#include "status.h" + +#define PASSWORD_LOCK_PROGRESS "/tmp/.passwordlock" + +static struct { + int checkfd; + //alarm_id_t alarm_id; /* -1 : None, others : set alarm */ + //Eina_Bool is_alarm; /* EINA_TRUE : can use alarm EINA_FALSE : cannot use */ + + int old_lock_type; + int lock_pid; + int lcd_state; + +#ifdef HAVE_X11 + lockw_data *lockw; +#endif +} s_lock_mgr = { + .checkfd = 0, + //.alarm_id = -1, + //.is_alarm = EINA_FALSE, + + .old_lock_type = 0, + .lock_pid = -1, + .lcd_state = -1, + +#ifdef HVAE_X11 + .lockw = NULL, +#endif +}; + + + +int lock_mgr_lcd_state_get(void) +{ + return s_lock_mgr.lcd_state; +} + + + +int lock_mgr_get_lock_pid(void) +{ + return s_lock_mgr.lock_pid; +} + + + +#if 0 +static int _alarm_del(alarm_id_t id, void * user_param) +{ + int ret = ALARMMGR_RESULT_SUCCESS; + + _D("delete alarm id : %d", id); + + ret = alarmmgr_remove_alarm(id); + if (ret != ALARMMGR_RESULT_SUCCESS) { + _E("Failed to remove alarm(%d)", ret ); + } + + return 0; +} + + + +static void _alarm_unset(void) +{ + int ret = ALARMMGR_RESULT_SUCCESS; + + ret = alarmmgr_enum_alarm_ids(_alarm_del, NULL); + if (ret != ALARMMGR_RESULT_SUCCESS) { + _E("Failed to get list of alarm ids"); + } +} + + + +static void _alarm_lockscreen_launch(alarm_id_t alarm_id, void *data) +{ + int ret = ALARMMGR_RESULT_SUCCESS; + + _D("alarm id : %d", alarm_id); + + /* launch lockscreen */ + if (!lock_mgr_lockscreen_launch()) { + _E("Failed to launch lockscreen"); + } + + if (alarm_id != -1) { + if (alarm_id != s_lock_mgr.alarm_id) { + _E("alarm ids are different callback->id(%d), s_lock_mgr.alarm_id(%d)", alarm_id, s_lock_mgr.alarm_id); + /* delete all registering alarm*/ + _alarm_unset(); + s_lock_mgr.alarm_id = -1; + } else { + ret = alarmmgr_remove_alarm(alarm_id); + if (ret != ALARMMGR_RESULT_SUCCESS) { + _E("Failed to remove alaram(%d)", ret); + /* delete all registering alarm*/ + _alarm_unset(); + } + s_lock_mgr.alarm_id = -1; + } + } +} + + + +static Eina_Bool _alarm_init(void) +{ + int ret = 0; + + /* alarm id initialize */ + s_lock_mgr.alarm_id = -1; + + ret = alarmmgr_init(PACKAGE_NAME); + if (ret != ALARMMGR_RESULT_SUCCESS) { + _E("Failed to initialize alarmmgr(%d)", ret); + return EINA_FALSE; + } + + ret = alarmmgr_set_cb((alarm_cb_t)_alarm_lockscreen_launch, NULL); + if (ret != ALARMMGR_RESULT_SUCCESS) { + _E("Failed to set cb func(%d)", ret); + return EINA_FALSE; + } + + _D("alarm init success"); + + return EINA_TRUE; +} +#endif + + + +void lock_mgr_sound_play(lock_sound_type_e type) +{ + int val = status_passive_get()->setappl_sound_lock_bool; + ret_if(!val); + + switch(type) { + case LOCK_SOUND_LOCK: + feedback_play_type(FEEDBACK_TYPE_SOUND, FEEDBACK_PATTERN_LOCK); + break; + case LOCK_SOUND_UNLOCK: + feedback_play_type(FEEDBACK_TYPE_SOUND, FEEDBACK_PATTERN_UNLOCK); + break; + case LOCK_SOUND_BTN_KEY: + feedback_play_type(FEEDBACK_TYPE_SOUND, FEEDBACK_PATTERN_SIP); + break; + case LOCK_SOUND_TAP: + feedback_play_type(FEEDBACK_TYPE_SOUND, FEEDBACK_PATTERN_TAP); + break; + default: + break; + } +} + + + +void lock_mgr_idle_lock_state_set(int lock_state) +{ + _D("lock state : %d", lock_state); + + int ret = 0; + + ret = vconf_set_int(VCONFKEY_IDLE_LOCK_STATE, lock_state); + if (ret < 0) { + _E("Failed to set vconfkey : VCONFKEY_IDLE_LOCK_STATE"); + } +} + + + +static void _after_launch_lock(int pid) +{ + int idle_lock_state = 0; + + if (dbus_util_send_oomadj(pid, OOM_ADJ_VALUE_DEFAULT) < 0) { + _E("cannot send oomadj for pid[%d]", pid); + } + process_mgr_set_lock_priority(pid); + display_unlock_state(LCD_OFF, PM_SLEEP_MARGIN); + s_lock_mgr.lock_pid = pid; + + idle_lock_state = status_passive_get()->idle_lock_state; + if (!idle_lock_state) { + lock_mgr_idle_lock_state_set(VCONFKEY_IDLE_LOCK); + lock_mgr_sound_play(LOCK_SOUND_LOCK); + } +} + + + +static int _lock_changed_cb(const char *appid, const char *key, const char *value, void *cfn, void *afn) +{ + _D("%s", __func__); + + return 0; +} + + + +static void _other_lockscreen_unlock(void) +{ + _D("unlock other lock screen"); + +#ifdef HAVE_X11 + window_mgr_unregister_event(s_lock_mgr.lockw); + window_mgr_fini(s_lock_mgr.lockw); + s_lock_mgr.lockw = NULL; +#endif +} + + + +#ifdef HAVE_X11 +static Eina_Bool _lock_create_cb(void *data, int type, void *event) +{ + _D("lockw(%p), lock_pid(%d)", s_lock_mgr.lockw, s_lock_mgr.lock_pid); + + if (window_mgr_set_effect(s_lock_mgr.lockw, s_lock_mgr.lock_pid, event) == EINA_TRUE) { + //FIXME sometimes show cb is not called. + if (window_mgr_set_prop(s_lock_mgr.lockw, s_lock_mgr.lock_pid, event) == EINA_FALSE) { + _E("window is not matched..!!"); + } + } + return ECORE_CALLBACK_PASS_ON; +} +#endif + + + +#ifdef HAVE_X11 +static Eina_Bool _lock_show_cb(void *data, int type, void *event) +{ + _D("lockw(%p), lock_pid(%d)", s_lock_mgr.lockw, s_lock_mgr.lock_pid); + + if (window_mgr_set_prop(s_lock_mgr.lockw, s_lock_mgr.lock_pid, event)) { + int lock_type = status_active_get()->setappl_screen_lock_type_int; + _D("lock type : %d", lock_type); + + window_mgr_set_scroll_prop(s_lock_mgr.lockw, lock_type); + } + + return ECORE_CALLBACK_CANCEL; +} +#endif + + + +void lock_mgr_unlock(void) +{ + int lock_type = status_active_get()->setappl_screen_lock_type_int; + _D("lock type(%d)", lock_type); + + lock_mgr_idle_lock_state_set(VCONFKEY_IDLE_UNLOCK); + lock_mgr_sound_play(LOCK_SOUND_UNLOCK); + s_lock_mgr.lock_pid = 0; + + if (lock_type == SETTING_SCREEN_LOCK_TYPE_OTHER) { + _other_lockscreen_unlock(); + } + +#ifdef HAVE_X11 + window_mgr_unregister_event(s_lock_mgr.lockw); +#endif +} + + + +static void _on_lcd_changed_receive(void *data, DBusMessage *msg) +{ + int lcd_on = 0; + int lcd_off = 0; + + _D("LCD signal is received"); + + lcd_on = dbus_message_is_signal(msg, DEVICED_INTERFACE_DISPLAY, MEMBER_LCD_ON); + lcd_off = dbus_message_is_signal(msg, DEVICED_INTERFACE_DISPLAY, MEMBER_LCD_OFF); + + if (lcd_on) { + _W("LCD on"); + s_lock_mgr.lcd_state = LCD_STATE_ON; + +#if 0 + /* delete all alarm registering */ + _D("delete alarm : id(%d)", s_lock_mgr.alarm_id); + _alarm_unset(); + s_lock_mgr.alarm_id = -1; + +#endif + } else if (lcd_off) { + s_lock_mgr.lcd_state = LCD_STATE_OFF; + + int idle_lock_state = status_passive_get()->idle_lock_state; + int lock_type = status_active_get()->setappl_screen_lock_type_int; + _D("idle_lock_state(%d), lock type(%d)", idle_lock_state, lock_type); + + if (!lock_mgr_lockscreen_launch()) { + _E("Failed to launch lockscreen"); + } + } else { + _E("%s dbus_message_is_signal error", DEVICED_INTERFACE_DISPLAY); + } +} + + + +Eina_Bool lock_mgr_lockscreen_launch(void) +{ + const char *lock_appid = NULL; + int lock_type = 0; + + lock_type = status_active_get()->setappl_screen_lock_type_int; + _D("lock type : %d", lock_type); + + //PM LOCK - don't go to sleep + display_lock_state(LCD_OFF, STAY_CUR_STATE, 0); + +#ifdef HAVE_X11 + /* reset window mgr before start win mgr */ + window_mgr_unregister_event(s_lock_mgr.lockw); + window_mgr_register_event(NULL, s_lock_mgr.lockw, _lock_create_cb, _lock_show_cb); +#endif + + lock_appid = status_passive_get()->setappl_3rd_lock_pkg_name_str; + if (!lock_appid) { + _E("set default lockscreen"); + lock_appid = STATUS_DEFAULT_LOCK_PKG_NAME; + } + + _D("lockscreen appid : %s", lock_appid); + + switch (lock_type) { + //case SETTING_SCREEN_LOCK_TYPE_NONE: + case SETTING_SCREEN_LOCK_TYPE_OTHER: + if (!strcmp(lock_appid, STATUS_DEFAULT_LOCK_PKG_NAME)) { + _D("ignore launching lockscreen"); + } else { + process_mgr_must_launch(lock_appid, NULL, NULL, _lock_changed_cb, _after_launch_lock); + //@TODO: need to check(add error popup) + } + break; + case SETTING_SCREEN_LOCK_TYPE_SWIPE: + process_mgr_must_launch(lock_appid, NULL, NULL, _lock_changed_cb, _after_launch_lock); + goto_if(s_lock_mgr.lock_pid < 0, ERROR); + break; + case SETTING_SCREEN_LOCK_TYPE_SIMPLE_PASSWORD: + case SETTING_SCREEN_LOCK_TYPE_PASSWORD: + process_mgr_must_launch(lock_appid, NULL, NULL, _lock_changed_cb, _after_launch_lock); + goto_if(s_lock_mgr.lock_pid < 0, ERROR); + + if (dbus_util_send_oomadj(s_lock_mgr.lock_pid, OOM_ADJ_VALUE_DEFAULT) < 0){ + _E("Failed to send oom dbus signal"); + } + + process_mgr_set_lock_priority(s_lock_mgr.lock_pid); + display_unlock_state(LCD_OFF, PM_SLEEP_MARGIN); + break; + default: + _E("type error(%d)", lock_type); + goto ERROR; + } + + _W("lock_pid : %d", s_lock_mgr.lock_pid); + + return EINA_TRUE; + +ERROR: + _E("Failed to launch lockscreen"); + + display_unlock_state(LCD_OFF, PM_SLEEP_MARGIN); + + return EINA_FALSE; +} + + + +static void _lock_daemon_init(void) +{ + _SECURE_D("default lock screen pkg name is %s", status_passive_get()->setappl_3rd_lock_pkg_name_str); + +#if 0 + /* init alarm manager */ + s_lock_mgr.is_alarm = _alarm_init(); +#endif + + /* register lcd changed cb */ + dbus_util_receive_lcd_status(_on_lcd_changed_receive, NULL); + +#ifdef HAVE_X11 + /* Create internal 1x1 window */ + s_lock_mgr.lockw = window_mgr_init(); +#endif +} + + + +static int _lock_type_changed_cb(status_active_key_e key, void *data) +{ + int lock_type = status_active_get()->setappl_screen_lock_type_int; + retv_if(lock_type == s_lock_mgr.old_lock_type, 1); + + _D("lock type is changed : %d -> %d", s_lock_mgr.old_lock_type, lock_type); + + s_lock_mgr.old_lock_type = lock_type; + + return 1; +} + + +int lock_mgr_daemon_start(void) +{ + int lock_type = 0; + int ret = 0; + + _lock_daemon_init(); + + lock_type = status_active_get()->setappl_screen_lock_type_int; + _D("lock type : %d", lock_type); + + ret = lock_mgr_lockscreen_launch(); + _D("ret : %d", ret); + + status_active_register_cb(STATUS_ACTIVE_KEY_SETAPPL_SCREEN_LOCK_TYPE_INT, _lock_type_changed_cb, NULL); + + if (feedback_initialize() != FEEDBACK_ERROR_NONE) { + _E("Failed to initialize feedback"); + } + + return ret; +} + + + +void lock_mgr_daemon_end(void) +{ +#ifdef HAVE_X11 + if (s_lock_mgr.lockw) { + free(s_lock_mgr.lockw); + } +#endif +} diff --git a/src/ivi/popup.c b/src/ivi/popup.c new file mode 100644 index 0000000..c52c055 --- /dev/null +++ b/src/ivi/popup.c @@ -0,0 +1,125 @@ +/* + * Copyright (c) 2000 - 2015 Samsung Electronics Co., Ltd. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include + +#include "popup.h" +#include "util.h" + +#define POPUP_DATA_KEY_WINDOW "__popup_window__" + +static void _popup_destroy(Evas_Object *popup) +{ + Evas_Object *win = NULL; + + if (popup) { + win = evas_object_data_del(popup, POPUP_DATA_KEY_WINDOW); + evas_object_del(popup); + + if (win) { + evas_object_del(win); + } + } +} + +static Evas_Object *_window_create(void) +{ + Evas_Object *win = NULL; + int win_w = 0, win_h = 0; + + win = elm_win_add(NULL, "STARTER-POPUP", ELM_WIN_DIALOG_BASIC); + retv_if(!win, NULL); + + elm_win_title_set(win, "STARTER-POPUP"); + elm_win_alpha_set(win, EINA_TRUE); + elm_win_borderless_set(win, EINA_TRUE); + elm_win_autodel_set(win, EINA_TRUE); + elm_win_raise(win); + + elm_win_screen_size_get(win, NULL, NULL, &win_w, &win_h); + _D("win size : (%dx%d)", win_w, win_h); + evas_object_resize(win, win_w, win_h); + + evas_object_show(win); + + return win; +} + + + +static void _popup_btn_clicked_cb(void *data, Evas_Object *obj, void *event_info) +{ + Evas_Object *popup = (Evas_Object *)data; + ret_if(!popup); + + _popup_destroy(popup); +} + + + +static void _popup_del_cb(void *data, Evas_Object *obj, void *event_info) +{ + ret_if(!obj); + + _popup_destroy(obj); +} + + + +Evas_Object *popup_create(const char *title, const char *text) +{ + Evas_Object *win = NULL; + Evas_Object *popup = NULL; + Evas_Object *btn = NULL; + + retv_if(!title, NULL); + retv_if(!text, NULL); + + win = _window_create(); + goto_if(!win, ERROR); + + popup = elm_popup_add(win); + goto_if(!popup, ERROR); + + elm_popup_orient_set(popup, ELM_POPUP_ORIENT_BOTTOM); + evas_object_size_hint_weight_set(popup, EVAS_HINT_EXPAND, EVAS_HINT_EXPAND); + elm_object_part_text_set(popup, "title,text", title); + elm_object_text_set(popup, text); + evas_object_data_set(popup, POPUP_DATA_KEY_WINDOW, win); + eext_object_event_callback_add(popup, EEXT_CALLBACK_BACK, _popup_del_cb, NULL); + evas_object_smart_callback_add(popup, "block,clicked", _popup_del_cb, NULL); + + /* ok button */ + btn = elm_button_add(popup); + goto_if(!btn, ERROR); + + elm_object_style_set(btn, "popup"); + elm_object_text_set(btn, S_("IDS_COM_SK_CONFIRM")); + elm_object_part_content_set(popup, "button1", btn); + evas_object_smart_callback_add(btn, "clicked", _popup_btn_clicked_cb, popup); + + evas_object_show(popup); + + return popup; + +ERROR: + _E("Failed to create popup"); + + _popup_destroy(popup); + + return NULL; +} + diff --git a/src/ivi/starter.c b/src/ivi/starter.c new file mode 100644 index 0000000..5bd63be --- /dev/null +++ b/src/ivi/starter.c @@ -0,0 +1,326 @@ +/* + * Copyright (c) 2000 - 2015 Samsung Electronics Co., Ltd. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include +#include + +#include "starter.h" +#include "lock_mgr.h" +#include "home_mgr.h" +#include "hw_key.h" +#include "process_mgr.h" +#include "util.h" +#include "status.h" +#include "hw_key.h" + +#define LOCKSCREEN_ENABLE 0 + +#define PWLOCK_LITE_PKG_NAME "org.tizen.pwlock-lite" + +#define DATA_UNENCRYPTED "unencrypted" +#define DATA_MOUNTED "mounted" +#define SD_DATA_ENCRYPTED "encrypted" +#define SD_CRYPT_META_FILE ".MetaEcfsFile" +#define MMC_MOUNT_POINT "/opt/storage/sdcard" + + + +static void _hide_home(void) +{ + int seq = status_active_get()->starter_sequence; + ret_if(seq == 1); + + vconf_set_int(VCONFKEY_STARTER_SEQUENCE, 0); +} + + + +static void _show_home(void) +{ + int show_menu = 0; + + if (status_active_get()->starter_sequence || !show_menu) { + vconf_set_int(VCONFKEY_STARTER_SEQUENCE, 1); + } +} + + + +#if 0 +static Eina_Bool _finish_boot_animation(void *data) +{ + if (vconf_set_int(VCONFKEY_BOOT_ANIMATION_FINISHED, 1) != 0) { + _E("Failed to set boot animation finished set"); + } + _show_home(); + + return ECORE_CALLBACK_CANCEL; +} + + + +static int _fail_to_launch_pwlock(const char *appid, const char *key, const char *value, void *cfn, void *afn) +{ + _finish_boot_animation(NULL); + return 0; +} + + + +static void _after_launch_pwlock(int pid) +{ + process_mgr_set_pwlock_priority(pid); + ecore_timer_add(0.5, _finish_boot_animation, NULL); +} +#endif + + + +static void _signal_handler(int signum, siginfo_t *info, void *unused) +{ + _D("_signal_handler : Terminated..."); + elm_exit(); +} + + + +static int _power_off_cb(status_active_key_e key, void *data) +{ + int val = status_active_get()->sysman_power_off_status; + + if (val == VCONFKEY_SYSMAN_POWER_OFF_DIRECT + || val == VCONFKEY_SYSMAN_POWER_OFF_RESTART) + { + _D("_power_off_cb : Terminated..."); + elm_exit(); + } + + return 1; +} + + + +static int _boot_animation_finished_cb(status_active_key_e key, void *data) +{ + int val = status_active_get()->boot_animation_finished; + _D("boot animation finished : %d", val); + + if (val == 1) { +#if LOCKSCREEN_ENABLE + lock_mgr_daemon_start(); +#endif + _show_home(); + } + + return 1; +} + + + +static void _language_changed_cb(keynode_t *node, void *data) +{ + char *lang = NULL; + + ret_if(!node); + + lang = vconf_keynode_get_str(node); + ret_if(!lang); + + _D("language is changed : %s", lang); + + elm_language_set(lang); +} + + + +static int _set_i18n(const char *domain, const char *dir) +{ + char *r = NULL; + + if (domain == NULL) { + errno = EINVAL; + return -1; + } + + char *lang = vconf_get_str(VCONFKEY_LANGSET); + r = setlocale(LC_ALL, lang); + if (!r) { + _E("setlocale() error"); + } + if (lang) { + free(lang); + } + + r = bindtextdomain(domain, dir); + if (!r) { + _E("bindtextdomain() error"); + } + + r = textdomain(domain); + if (!r) { + _E("textdomain() error"); + } + + if (vconf_notify_key_changed(VCONFKEY_LANGSET, _language_changed_cb, NULL) < 0) { + _E("Failed to register changed cb : %s", VCONFKEY_LANGSET); + } + + return 0; +} + + + +static int _check_dead_signal(int pid, void *data) +{ + int home_pid = 0; +#if LOCKSCREEN_ENABLE + int volume_pid = 0; + int lock_pid = 0; +#endif + + _D("Process %d is termianted", pid); + + if (pid < 0) { + _E("pid : %d", pid); + return 0; + } + + home_pid = home_mgr_get_home_pid(); +#if LOCKSCREEN_ENABLE + volume_pid = home_mgr_get_volume_pid(); + lock_pid = lock_mgr_get_lock_pid(); +#endif + + if (pid == home_pid) { + _D("Homescreen is dead"); + home_mgr_relaunch_homescreen(); +#if LOCKSCREEN_ENABLE + } else if (pid == volume_pid) { + _D("volume is dead"); + home_mgr_relaunch_volume(); + } else if (pid == lock_pid) { + _D("lockscreen is dead"); + lock_mgr_unlock(); +#endif + } else { + _D("Unknown process, ignore it"); + } + + return 0; +} + + + +static void _init(struct appdata *ad) +{ + struct sigaction act; + + memset(&act,0x00,sizeof(struct sigaction)); + act.sa_sigaction = _signal_handler; + act.sa_flags = SA_SIGINFO; + + int ret = sigemptyset(&act.sa_mask); + if (ret < 0) { + _E("Failed to sigemptyset[%s]", strerror(errno)); + } + ret = sigaddset(&act.sa_mask, SIGTERM); + if (ret < 0) { + _E("Failed to sigaddset[%s]", strerror(errno)); + } + ret = sigaction(SIGTERM, &act, NULL); + if (ret < 0) { + _E("Failed to sigaction[%s]", strerror(errno)); + } + + _set_i18n(PACKAGE, LOCALEDIR); + + status_register(); + status_active_register_cb(STATUS_ACTIVE_KEY_SYSMAN_POWER_OFF_STATUS, _power_off_cb, NULL); + status_active_register_cb(STATUS_ACTIVE_KEY_BOOT_ANIMATION_FINISHED, _boot_animation_finished_cb, NULL); + + /* Ordering : _hide_home -> process_mgr_must_launch(pwlock) -> _show_home */ + _hide_home(); +#if 0 + process_mgr_must_launch(PWLOCK_LITE_PKG_NAME, NULL, NULL, _fail_to_launch_pwlock, _after_launch_pwlock); +#endif + + hw_key_create_window(); + home_mgr_init(NULL); + + aul_listen_app_dead_signal(_check_dead_signal, NULL); +} + + + +static void _fini(struct appdata *ad) +{ + home_mgr_fini(); + hw_key_destroy_window(); +#if LOCKSCREEN_ENABLE + lock_mgr_daemon_end(); +#endif + + status_active_unregister_cb(STATUS_ACTIVE_KEY_SYSMAN_POWER_OFF_STATUS, _power_off_cb); + status_active_unregister_cb(STATUS_ACTIVE_KEY_BOOT_ANIMATION_FINISHED, _boot_animation_finished_cb); + status_unregister(); + + if (vconf_ignore_key_changed(VCONFKEY_LANGSET, _language_changed_cb) < 0) { + _E("Failed to unregister changed cb : %s", VCONFKEY_LANGSET); + } +} + + + +int main(int argc, char *argv[]) +{ + struct appdata ad; + int ret = 0; + + _D("starter is launched..!!"); + + ret = elm_init(argc, argv); + if (ret != 1) { + _E("elm_init() failed : %d", ret); + return -1; + } + + ret = ecore_wl_init(NULL); + if (ret == 0) { + _E("ecore_wl_init() failed : %d", ret); + elm_shutdown(); + return -1; + } + + _init(&ad); + + elm_run(); + + _fini(&ad); + elm_shutdown(); + + return 0; +} diff --git a/src/ivi/window_mgr.c b/src/ivi/window_mgr.c new file mode 100644 index 0000000..2f1a198 --- /dev/null +++ b/src/ivi/window_mgr.c @@ -0,0 +1,377 @@ +/* + * Copyright (c) 2000 - 2015 Samsung Electronics Co., Ltd. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include +#include +#include +#include +#include + +#ifdef HAVE_X11 +#include +#include +#include +#endif + +#include "window_mgr.h" +#include "util.h" + +#define STR_ATOM_PANEL_SCROLLABLE_STATE "_E_MOVE_PANEL_SCROLLABLE_STATE" + + + +#ifdef HAVE_X11 +struct _lockw_data { + Eina_Bool is_registered; + Ecore_X_Window lock_x_window; + + Ecore_Event_Handler *h_wincreate; + Ecore_Event_Handler *h_winshow; + Ecore_Event_Handler *h_winhide; +}; + + + +static int _is_on_screen(Ecore_X_Display * dpy, Ecore_X_Window window) +{ + Ecore_X_Window root; + Window child; + Window win; + + int rel_x = 0; + int rel_y = 0; + int abs_x = 0; + int abs_y = 0; + + unsigned int width = 0; + unsigned int height = 0; + unsigned int border = 0; + unsigned int depth = 0; + unsigned int root_w = 0; + unsigned int root_h = 0; + + Eina_Bool ret = FALSE; + + root = ecore_x_window_root_first_get(); + XGetGeometry(dpy, root, &win, &rel_x, &rel_y, &root_w, &root_h, &border, &depth); + _D("root rel_x[%d] rel_y[%d] border[%d] width[%d] height[%d]", rel_x, rel_y, border, root_w, root_h); + + if (XGetGeometry(dpy, window, &win, &rel_x, &rel_y, &width, &height, &border, &depth)) { + if (XTranslateCoordinates(dpy, window, root, 0, 0, &abs_x, &abs_y, &child)) { + _D("abs_x[%d] abs_y[%d] border[%d] width[%d] height[%d]", abs_x, abs_y, border, width, height); + if ((abs_x - border) >= root_w + || (abs_y - border) >= root_h + || (width + abs_x) <= 0 || (height + abs_y) <= 0) + { + ret = FALSE; + } else { + ret = (width == root_w) && (height == root_h); + } + } + } + + return ret; +} + + + +static Window _get_user_created_window(Window win) +{ + Atom type_ret = 0; + int ret, size_ret = 0; + unsigned long num_ret = 0, bytes = 0; + unsigned char *prop_ret = NULL; + unsigned int xid; + Atom prop_user_created_win; + + prop_user_created_win = XInternAtom(ecore_x_display_get(), "_E_USER_CREATED_WINDOW", False); + + ret = XGetWindowProperty(ecore_x_display_get() + , win, prop_user_created_win + , 0L, 1L + , False, 0 + , &type_ret, &size_ret + , &num_ret, &bytes + , &prop_ret); + if (ret != Success) { + if (prop_ret) + XFree((void *) prop_ret); + return win; + } else if (!prop_ret) { + return win; + } + + memcpy(&xid, prop_ret, sizeof(unsigned int)); + XFree((void *)prop_ret); + + return xid; + +} + + + +int window_mgr_get_focus_window_pid(void) +{ + Ecore_X_Window x_win_focused = 0; + int pid = 0; + int ret = -1; + + _D("%s, %d", __func__, __LINE__); + + x_win_focused = ecore_x_window_focus_get(); + ret = ecore_x_netwm_pid_get(x_win_focused, &pid); + if(ret != 1) { + _E("Can't get pid for focus x window (%x)\n", x_win_focused); + return -1; + } + _D("PID(%d) for focus x window (%x)\n", pid, x_win_focused); + + return pid; +} + + + +static void _pwd_transient_set(Ecore_X_Window win, Ecore_X_Window for_win) +{ + _W("%p is transient for %p", win, for_win); + + ecore_x_icccm_transient_for_set(win, for_win); +} + + + +static void _pwd_transient_unset(Ecore_X_Window xwin) +{ + ret_if(!xwin); + + _W("%p is not transient", xwin); + ecore_x_icccm_transient_for_unset(xwin); +} + + + +Eina_Bool window_mgr_pwd_transient_set(void *data) +{ + Evas_Object *pwd_win = NULL; + Ecore_X_Window pwd_x_win; + lockw_data *lockw = (lockw_data *) data; + retv_if(!lockw, EINA_FALSE); + + pwd_win = lock_pwd_util_win_get(); + retv_if(!pwd_win, EINA_FALSE); + + pwd_x_win = elm_win_xwindow_get(pwd_win); + retv_if(!pwd_x_win, EINA_FALSE); + + retv_if(!lockw->lock_x_window, EINA_FALSE); + + /* unset transient */ + _pwd_transient_unset(lockw->lock_x_window); + + /* set transient */ + _pwd_transient_set(lockw->lock_x_window, pwd_x_win); + + return EINA_TRUE; +} + + + +Eina_Bool window_mgr_set_prop(lockw_data * data, int lock_app_pid, void *event) +{ + Ecore_X_Event_Window_Create *e = event; + Ecore_X_Window user_window = 0; + lockw_data *lockw = (lockw_data *) data; + int pid = 0; + int ret = 0; + + retv_if(!lockw, EINA_FALSE); + + user_window = _get_user_created_window((Window) (e->win)); + + ret = ecore_x_netwm_pid_get(user_window, &pid); + retv_if(ret != 1, EINA_FALSE); + + _D("Check PID(%d) window. (lock_app_pid : %d)", pid, lock_app_pid); + + if (lock_app_pid == pid) { + if (_is_on_screen(ecore_x_display_get(), user_window) == TRUE) { + lockw->lock_x_window = user_window; + /* window effect : fade in /out */ + ecore_x_icccm_name_class_set(user_window, "LOCK_SCREEN", "LOCK_SCREEN"); + ecore_x_netwm_window_type_set(user_window, ECORE_X_WINDOW_TYPE_NOTIFICATION); + utilx_set_system_notification_level(ecore_x_display_get(), user_window, UTILX_NOTIFICATION_LEVEL_NORMAL); + utilx_set_window_opaque_state(ecore_x_display_get(), user_window, UTILX_OPAQUE_STATE_ON); + + /* set transient */ + if (!window_mgr_pwd_transient_set(lockw)) { + _E("Failed to set transient"); + } + + return EINA_TRUE; + } + } + return EINA_FALSE; +} + + + +Eina_Bool window_mgr_set_effect(lockw_data * data, int lock_app_pid, void *event) +{ + Ecore_X_Event_Window_Create *e = event; + Ecore_X_Window user_window = 0; + int pid = 0; + int ret = 0; + + user_window = _get_user_created_window((Window) (e->win)); + ret = ecore_x_netwm_pid_get(user_window, &pid); + retv_if(ret != 1, EINA_FALSE); + + if (lock_app_pid == pid) { + if (_is_on_screen(ecore_x_display_get(), user_window) == TRUE) { + Ecore_X_Atom ATOM_WINDOW_EFFECT_ENABLE = 0; + unsigned int effect_state = 0; + + ATOM_WINDOW_EFFECT_ENABLE = ecore_x_atom_get("_NET_CM_WINDOW_EFFECT_ENABLE"); + if (ATOM_WINDOW_EFFECT_ENABLE) { + ecore_x_window_prop_card32_set(user_window, ATOM_WINDOW_EFFECT_ENABLE, &effect_state, 1); + } else { + _E("ecore_x_atom_get() failed"); + } + return EINA_TRUE; + } + } + return EINA_FALSE; +} + + + +void window_mgr_set_scroll_prop(lockw_data *data, int lock_type) +{ + lockw_data *lockw = (lockw_data *) data; + Ecore_X_Atom ATOM_PANEL_SCROLLABLE_STATE = 0; + unsigned int val[3] = { 0, }; + + ret_if(!lockw); + + ATOM_PANEL_SCROLLABLE_STATE = ecore_x_atom_get(STR_ATOM_PANEL_SCROLLABLE_STATE); + if (lock_type == SETTING_SCREEN_LOCK_TYPE_SIMPLE_PASSWORD || + lock_type == SETTING_SCREEN_LOCK_TYPE_PASSWORD) { + val[0] = 0; // always enable F + val[1] = 0; // quickpanel enable F + val[2] = 0; // apptray enable F + } else { + val[0] = 0; // always enable F + val[1] = 1; // quickpanel enable T + val[2] = 0; // apptray enable F + } + ecore_x_window_prop_card32_set(lockw->lock_x_window, ATOM_PANEL_SCROLLABLE_STATE, val, 3); +} + + + +void window_mgr_register_event(void *data, lockw_data * lockw, + Eina_Bool (*create_cb) (void *, int, void *), + Eina_Bool (*show_cb) (void *, int, void *)) +{ + Ecore_X_Window root_window; + + ret_if(!lockw); + + if (lockw->is_registered) { + _E("Already register event cb"); + return; + } + + /* For getting window x event */ + root_window = ecore_x_window_root_first_get(); + ecore_x_window_client_sniff(root_window); + + lockw->h_wincreate = ecore_event_handler_add(ECORE_X_EVENT_WINDOW_CREATE, create_cb, data); + lockw->h_winshow = ecore_event_handler_add(ECORE_X_EVENT_WINDOW_SHOW, show_cb, data); + + lockw->is_registered = EINA_TRUE; +} + + + +static inline void _unregister_event(lockw_data *lockw) +{ + Ecore_X_Window root_window; + + /* unset transient */ + _pwd_transient_unset(lockw->lock_x_window); + + /* delete getting window x event */ + root_window = ecore_x_window_root_first_get(); + ecore_x_window_client_sniff(root_window); + + /* delete window create event handler */ + if (lockw->h_wincreate) { + ecore_event_handler_del(lockw->h_wincreate); + lockw->h_wincreate = NULL; + } + if (lockw->h_winshow) { + ecore_event_handler_del(lockw->h_winshow); + lockw->h_winshow = NULL; + } + if (lockw->h_winhide) { + ecore_event_handler_del(lockw->h_winhide); + lockw->h_winhide = NULL; + } + + ecore_x_pointer_ungrab(); + + lockw->is_registered = EINA_FALSE; +} + + + +void window_mgr_unregister_event(lockw_data *lockw) +{ + ret_if(!lockw); + + if (!lockw->is_registered) { + _E("event cb is not registered"); + return; + } + + _unregister_event(lockw); +} + + + +lockw_data *window_mgr_init(void) +{ + lockw_data *lockw = NULL; + + lockw = calloc(1, sizeof(*lockw)); + + return lockw; +} + + + +void window_mgr_fini(lockw_data *lockw) +{ + ret_if(!lockw); + + if (lockw->is_registered) { + _unregister_event(lockw); + } + + free(lockw); +} +#endif