9 #include <Elementary.h>
14 #include <vconf-keys.h>
20 #define APP_NAME "org.tizen.answer"
21 #define BUS_NAME "org.tizen.dialer"
23 #define IFACE "org.tizen.dialer.Control"
25 static int _log_domain = -1;
26 #define ERR(...) EINA_LOG_DOM_ERR(_log_domain, __VA_ARGS__)
27 #define INF(...) EINA_LOG_DOM_INFO(_log_domain, __VA_ARGS__)
28 #define DBG(...) EINA_LOG_DOM_DBG(_log_domain, __VA_ARGS__)
30 static E_DBus_Connection *bus_conn = NULL;
32 typedef struct _Call {
33 const char *line_id, *img, *type, *name;
36 typedef struct _Call_Screen {
37 Evas_Object *win, *layout;
39 Eina_Bool screen_visible;
40 E_DBus_Signal_Handler *sig_call_add;
41 E_DBus_Signal_Handler *sig_call_removed;
42 E_DBus_Signal_Handler *sig_name_changed;
45 static void _set_notification_type(Evas_Object *win, Eina_Bool high)
51 static Eina_Bool _phone_locked(void)
55 if (vconf_get_int(VCONFKEY_IDLE_LOCK_STATE, &lock) == -1)
58 if (lock == VCONFKEY_IDLE_LOCK)
63 return EINA_TRUE; /* always have it to show */
67 static Call *_call_create(DBusMessage *msg)
70 const char *img, *name, *id, *type;
73 dbus_error_init(&err);
74 dbus_message_get_args(msg, &err, DBUS_TYPE_STRING, &img,
75 DBUS_TYPE_STRING, &id, DBUS_TYPE_STRING,
76 &name, DBUS_TYPE_STRING, &type,
79 if (dbus_error_is_set(&err)) {
80 ERR("Could not parse message: %s: %s", err.name,
84 call = calloc(1, sizeof(Call));
85 EINA_SAFETY_ON_NULL_RETURN_VAL(call, NULL);
86 call->img = eina_stringshare_add(img);
87 call->line_id = eina_stringshare_add(id);
88 call->type = eina_stringshare_add(type);
89 call->name = eina_stringshare_add(name);
90 DBG("c=%p line_id=%s, name=%s, type=%s, img=%s",
91 call, call->line_id, call->name, call->type, call->img);
95 static void _call_destroy(Call *c)
97 DBG("c=%p line_id=%s, name=%s, type=%s, img=%s",
98 c, c->line_id, c->name, c->type, c->img);
99 eina_stringshare_del(c->img);
100 eina_stringshare_del(c->line_id);
101 eina_stringshare_del(c->type);
102 eina_stringshare_del(c->name);
106 static void _call_screen_show(Call_Screen *cs)
109 Evas_Object *icon = NULL;
111 INF("Show line_id=%s, name=%s, type=%s",
112 c->line_id, c->name, c->type);
114 _set_notification_type(cs->win, EINA_TRUE);
115 cs->screen_visible = EINA_TRUE;
116 elm_win_raise(cs->win);
117 elm_win_activate(cs->win);
119 if (strcmp("", c->name) == 0)
120 elm_object_part_text_set(cs->layout, "elm.text.name",
123 elm_object_part_text_set(cs->layout, "elm.text.name",
125 icon = elm_icon_add(cs->layout);
127 if (strcmp("", c->img) != 0) {
128 elm_image_file_set(icon, c->img, NULL);
130 elm_icon_standard_set(icon, "no-picture");
132 elm_object_part_text_set(cs->layout, "elm.text.state", _("Incoming..."));
133 elm_object_part_text_set(cs->layout, "elm.text.phone.type", c->type);
134 elm_object_part_content_set(cs->layout, "elm.swallow.photo", icon);
135 elm_object_signal_emit(cs->layout, "show,activecall", "gui");
136 evas_object_show(cs->win);
138 //screen can't be off
139 power_lock_state(POWER_STATE_NORMAL, 0);
143 static void _sig_call_added(void *data, DBusMessage *msg)
145 Call_Screen *cs = data;
146 Eina_Bool locked = _phone_locked();
148 DBG("previous=%p, visible=%d, locked=%d", cs->call, cs->screen_visible,
152 _call_destroy(cs->call);
153 cs->call = _call_create(msg);
158 _call_screen_show(cs);
161 static void _call_screen_hide(Call_Screen *cs)
164 cs->screen_visible = EINA_FALSE;
165 _set_notification_type(cs->win, EINA_FALSE);
166 elm_object_signal_emit(cs->layout, "hide,activecall", "gui");
167 evas_object_hide(cs->win);
169 //screen can go off again
170 power_unlock_state(POWER_STATE_NORMAL);
174 static void _sig_call_removed(void *data, DBusMessage *msg __UNUSED__)
176 Call_Screen *cs = data;
178 DBG("previous=%p, visible=%d", cs->call, cs->screen_visible);
181 _call_destroy(cs->call);
185 if (!cs->screen_visible)
188 _call_screen_hide(cs);
191 static void _signal_handlers_add(Call_Screen *cs)
193 cs->sig_call_add = e_dbus_signal_handler_add(bus_conn, BUS_NAME,
198 cs->sig_call_removed = e_dbus_signal_handler_add(bus_conn,
206 static void _signal_handlers_del(Call_Screen *cs)
208 if (cs->sig_call_add) {
209 e_dbus_signal_handler_del(bus_conn, cs->sig_call_add);
210 cs->sig_call_add = NULL;
213 if (cs->sig_call_removed) {
214 e_dbus_signal_handler_del(bus_conn, cs->sig_call_removed);
215 cs->sig_call_removed = NULL;
219 static void _daemon_method_call_reply(void *data __UNUSED__,
220 DBusMessage *msg __UNUSED__,
223 if (dbus_error_is_set(err)) {
224 ERR("Error calling remote method: %s %s", err->name,
229 static void _daemon_method_call_make_with_reply(const char *method,
230 E_DBus_Method_Return_Cb cb,
235 msg = dbus_message_new_method_call(BUS_NAME, PATH, IFACE, method);
236 INF("msg=%p name=%s, path=%s, %s.%s()",
237 msg, BUS_NAME, PATH, IFACE, method);
238 EINA_SAFETY_ON_NULL_RETURN(msg);
240 e_dbus_message_send(bus_conn, msg, cb, -1, data);
241 dbus_message_unref(msg);
244 static void _daemon_method_call_make(const char *method)
246 _daemon_method_call_make_with_reply(method, _daemon_method_call_reply,
250 static void _daemon_available_call_reply(void *data, DBusMessage *msg,
253 Call_Screen *cs = data;
255 if (dbus_error_is_set(err)) {
256 const char *err_name = "org.tizen.dialer.error.NotAvailable";
257 if (strcmp(err_name, err->name) == 0) {
258 DBG("no incoming calls");
262 ERR("Error calling remote method: %s %s", err->name,
268 _call_destroy(cs->call);
270 cs->call = _call_create(msg);
271 _call_screen_show(cs);
274 static void _service_start_reply(void *data, DBusMessage *msg __UNUSED__,
277 Call_Screen *cs = data;
278 if (dbus_error_is_set(err)) {
279 ERR("Error: %s %s", err->name, err->message);
282 _signal_handlers_add(cs);
283 _daemon_method_call_make_with_reply("GetAvailableCall",
284 _daemon_available_call_reply,
288 static void _has_owner_cb(void *data, DBusMessage *msg __UNUSED__,
293 Call_Screen *cs = data;
296 if (dbus_error_is_set(error)) {
297 ERR("Error: %s %s", error->name, error->message);
300 dbus_error_init(&err);
301 dbus_message_get_args(msg, &err, DBUS_TYPE_BOOLEAN, &online,
305 INF("no dialer, start it");
306 e_dbus_start_service_by_name(bus_conn, BUS_NAME, 0,
307 _service_start_reply, cs);
311 _signal_handlers_add(cs);
312 _daemon_method_call_make_with_reply("GetAvailableCall",
313 _daemon_available_call_reply,
317 static void _name_owner_changed(void *data, DBusMessage *msg)
319 Call_Screen *cs = data;
321 const char *name, *from, *to;
323 dbus_error_init(&err);
324 if (!dbus_message_get_args(msg, &err,
325 DBUS_TYPE_STRING, &name,
326 DBUS_TYPE_STRING, &from,
327 DBUS_TYPE_STRING, &to,
328 DBUS_TYPE_INVALID)) {
329 ERR("Could not get NameOwnerChanged arguments: %s: %s",
330 err.name, err.message);
331 dbus_error_free(&err);
335 if (strcmp(name, BUS_NAME) != 0)
338 if ((to == NULL) || (*to == '\0')) {
339 INF("missed dialer");
340 _signal_handlers_del(cs);
344 INF("got new dialer at %s", to);
345 _signal_handlers_add(cs);
346 _daemon_method_call_make_with_reply("GetAvailableCall",
347 _daemon_available_call_reply,
351 static Eina_Bool _dbus_init(Call_Screen *cs)
354 char *bus_name = BUS_NAME;
356 INF("Running on Session bus");
357 bus_conn = e_dbus_bus_get(DBUS_BUS_SESSION);
360 ERR("Could not fetch the DBus session");
364 cs->sig_name_changed = e_dbus_signal_handler_add(
365 bus_conn, E_DBUS_FDO_BUS, E_DBUS_FDO_PATH, E_DBUS_FDO_INTERFACE,
366 "NameOwnerChanged", _name_owner_changed, cs);
368 msg = dbus_message_new_method_call(E_DBUS_FDO_BUS, E_DBUS_FDO_PATH,
369 E_DBUS_FDO_INTERFACE,
372 EINA_SAFETY_ON_NULL_RETURN_VAL(msg, EINA_FALSE);
374 if (!dbus_message_append_args(msg, DBUS_TYPE_STRING, &bus_name,
378 e_dbus_message_send(bus_conn, msg, _has_owner_cb, -1, cs);
379 dbus_message_unref(msg);
383 dbus_message_unref(msg);
387 Evas_Object *gui_layout_add(Evas_Object *parent, const char *style)
389 Evas_Object *layout = elm_layout_add(parent);
390 if (!elm_layout_theme_set(layout, "layout", "dialer", style)) {
391 evas_object_del(layout);
398 static int _running_apps_iter(const aul_app_info *info, void *data __UNUSED__)
400 if (strcmp(info->pkg_name, "org.tizen.draglock") == 0)
401 aul_terminate_pid(info->pid);
406 static void _phone_unlock_screen(void)
408 INF("unlock screen");
410 aul_app_get_running_app_info(_running_apps_iter, NULL);
414 static void _slider_pos_changed_cb(void *data, Evas_Object *obj,
415 void *event_inf __UNUSED__)
417 Call_Screen *cs = data;
418 const char *label = elm_actionslider_selected_label_get(obj);
423 if (strcmp(label, "Answer") == 0) {
424 _daemon_method_call_make("AnswerCall");
425 elm_actionslider_indicator_pos_set(obj,
426 ELM_ACTIONSLIDER_CENTER);
427 _phone_unlock_screen();
428 _call_screen_hide(cs);
429 } else if (strcmp(label, "Hangup") == 0) {
430 _daemon_method_call_make("HangupCall");
431 elm_actionslider_indicator_pos_set(obj,
432 ELM_ACTIONSLIDER_CENTER);
436 static Eina_Bool _gui_init(Call_Screen *cs)
438 Evas_Object *win, *obj, *lay, *slider;
440 char def_theme[PATH_MAX];
441 /* should never, ever quit */
443 elm_policy_set(ELM_POLICY_QUIT, ELM_POLICY_QUIT_NONE);
444 elm_app_compile_bin_dir_set(PACKAGE_BIN_DIR);
445 elm_app_compile_data_dir_set(PACKAGE_DATA_DIR);
446 elm_app_info_set(_gui_init, "answer-ofono-efl", "themes/default.edj");
448 snprintf(def_theme, sizeof(def_theme), "%s/themes/default.edj",
449 elm_app_data_dir_get());
451 elm_theme_extension_add(NULL, def_theme);
452 elm_theme_overlay_add(NULL, def_theme);
454 win = elm_win_util_standard_add("answer screen", "oFono Answer");
455 EINA_SAFETY_ON_NULL_RETURN_VAL(win, EINA_FALSE);
456 elm_win_autodel_set(win, EINA_FALSE);
457 elm_object_part_text_set(win, "elm.text.name", _("Unknown"));
458 elm_object_part_text_set(win, "elm.text.state", _("Incoming..."));
460 lay = gui_layout_add(win, "answer");
461 EINA_SAFETY_ON_NULL_RETURN_VAL(lay, EINA_FALSE);
462 evas_object_show(lay);
464 slider = elm_actionslider_add(win);
465 EINA_SAFETY_ON_NULL_GOTO(slider, err_obj);
466 elm_object_style_set(slider, "answer");
467 elm_actionslider_indicator_pos_set(slider, ELM_ACTIONSLIDER_CENTER);
468 elm_actionslider_magnet_pos_set(slider, ELM_ACTIONSLIDER_CENTER);
469 elm_object_part_text_set(slider, "left", _("Hangup"));
470 elm_object_part_text_set(slider, "right", _("Answer"));
471 evas_object_smart_callback_add(slider, "selected",
472 _slider_pos_changed_cb, cs);
473 evas_object_show(slider);
474 elm_object_part_content_set(lay, "elm.swallow.slider", slider);
476 obj = elm_layout_edje_get(lay);
477 edje_object_size_min_get(obj, &w, &h);
478 if ((w == 0) || (h == 0))
479 edje_object_size_min_restricted_calc(obj, &w, &h, w, h);
480 if ((w == 0) || (h == 0))
481 edje_object_parts_extends_calc(obj, NULL, NULL, &w, &h);
482 evas_object_resize(lay, w, h);
483 evas_object_resize(win, w, h);
490 evas_object_del(win);
494 EAPI int elm_main(int argc __UNUSED__, char **argv __UNUSED__)
498 if (!elm_need_e_dbus()) {
499 ERR("Could not start E_dbus");
503 _log_domain = eina_log_domain_register("answer_daemon", NULL);
504 cs = calloc(1, sizeof(Call_Screen));
505 EINA_SAFETY_ON_NULL_RETURN_VAL(cs, -1);
507 EINA_SAFETY_ON_FALSE_GOTO(_dbus_init(cs), err_dbus);
508 EINA_SAFETY_ON_FALSE_GOTO(_gui_init(cs), err_dbus);
512 _signal_handlers_del(cs);
514 if (cs->sig_name_changed)
515 e_dbus_signal_handler_del(bus_conn, cs->sig_name_changed);
517 evas_object_del(cs->win);
519 _call_destroy(cs->call);