efl/src/lib/elementary/elm_clock.c

967 lines
27 KiB
C

#ifdef HAVE_CONFIG_H
# include "elementary_config.h"
#endif
#define EFL_ACCESS_OBJECT_PROTECTED
#define EFL_UI_FOCUS_COMPOSITION_PROTECTED
#define EFL_UI_FOCUS_COMPOSITION_ADAPTER_PROTECTED
#include <Elementary.h>
#include "elm_priv.h"
#include "elm_widget_clock.h"
#include "efl_ui_focus_composition_adapter.eo.h"
#define MY_CLASS ELM_CLOCK_CLASS
#define MY_CLASS_NAME "Elm_Clock"
#define MY_CLASS_NAME_LEGACY "elm_clock"
#define DEFAULT_FIRST_INTERVAL 0.85
static void _time_update(Evas_Object *obj, Eina_Bool theme_update);
static const char SIG_CHANGED[] = "changed";
static const Evas_Smart_Cb_Description _smart_callbacks[] = {
{SIG_CHANGED, ""},
{SIG_WIDGET_LANG_CHANGED, ""}, /**< handled by elm_widget */
{SIG_WIDGET_ACCESS_CHANGED, ""}, /**< handled by elm_widget */
{SIG_LAYOUT_FOCUSED, ""}, /**< handled by elm_layout */
{SIG_LAYOUT_UNFOCUSED, ""}, /**< handled by elm_layout */
{NULL, NULL}
};
static Eina_Bool
_on_clock_val_up(void *data)
{
ELM_CLOCK_DATA_GET(data, sd);
if (!sd->edit) goto clock_val_up_cancel;
if (!sd->sel_obj) goto clock_val_up_cancel;
if (sd->sel_obj == sd->digit[0])
{
sd->hrs = sd->hrs + 12;
if (sd->hrs >= 24) sd->hrs -= 24;
}
if (sd->sel_obj == sd->digit[1])
{
sd->hrs = sd->hrs + 1;
if (sd->hrs >= 24) sd->hrs -= 24;
}
if (sd->sel_obj == sd->digit[2])
{
sd->min = sd->min + 10;
if (sd->min >= 60) sd->min -= 60;
}
if (sd->sel_obj == sd->digit[3])
{
sd->min = sd->min + 1;
if (sd->min >= 60) sd->min -= 60;
}
if (sd->sel_obj == sd->digit[4])
{
sd->sec = sd->sec + 10;
if (sd->sec >= 60) sd->sec -= 60;
}
if (sd->sel_obj == sd->digit[5])
{
sd->sec = sd->sec + 1;
if (sd->sec >= 60) sd->sec -= 60;
}
if (sd->sel_obj == sd->am_pm_obj)
{
sd->hrs = sd->hrs + 12;
if (sd->hrs > 23) sd->hrs -= 24;
}
sd->interval = sd->interval / 1.05;
ecore_timer_interval_set(sd->spin, sd->interval);
_time_update(data, EINA_FALSE);
efl_event_callback_legacy_call(data, ELM_CLOCK_EVENT_CHANGED, NULL);
return ECORE_CALLBACK_RENEW;
clock_val_up_cancel:
sd->spin = NULL;
return ECORE_CALLBACK_CANCEL;
}
static Eina_Bool
_on_clock_val_down(void *data)
{
ELM_CLOCK_DATA_GET(data, sd);
if (!sd->edit) goto clock_val_down_cancel;
if (!sd->sel_obj) goto clock_val_down_cancel;
if (sd->sel_obj == sd->digit[0])
{
sd->hrs = sd->hrs - 12;
if (sd->hrs < 0) sd->hrs += 24;
}
if (sd->sel_obj == sd->digit[1])
{
sd->hrs = sd->hrs - 1;
if (sd->hrs < 0) sd->hrs += 24;
}
if (sd->sel_obj == sd->digit[2])
{
sd->min = sd->min - 10;
if (sd->min < 0) sd->min += 60;
}
if (sd->sel_obj == sd->digit[3])
{
sd->min = sd->min - 1;
if (sd->min < 0) sd->min += 60;
}
if (sd->sel_obj == sd->digit[4])
{
sd->sec = sd->sec - 10;
if (sd->sec < 0) sd->sec += 60;
}
if (sd->sel_obj == sd->digit[5])
{
sd->sec = sd->sec - 1;
if (sd->sec < 0) sd->sec += 60;
}
if (sd->sel_obj == sd->am_pm_obj)
{
sd->hrs = sd->hrs - 12;
if (sd->hrs < 0) sd->hrs += 24;
}
sd->interval = sd->interval / 1.05;
ecore_timer_interval_set(sd->spin, sd->interval);
_time_update(data, EINA_FALSE);
efl_event_callback_legacy_call(data, ELM_CLOCK_EVENT_CHANGED, NULL);
return ECORE_CALLBACK_RENEW;
clock_val_down_cancel:
sd->spin = NULL;
return ECORE_CALLBACK_CANCEL;
}
static void
_on_clock_val_up_start(void *data,
Evas_Object *obj,
const char *emission EINA_UNUSED,
const char *source EINA_UNUSED)
{
ELM_CLOCK_DATA_GET(data, sd);
sd->interval = sd->first_interval;
sd->sel_obj = obj;
ecore_timer_del(sd->spin);
sd->spin = ecore_timer_add(sd->interval, _on_clock_val_up, data);
_on_clock_val_up(data);
}
static void
_on_clock_val_down_start(void *data,
Evas_Object *obj,
const char *emission EINA_UNUSED,
const char *source EINA_UNUSED)
{
ELM_CLOCK_DATA_GET(data, sd);
sd->interval = sd->first_interval;
sd->sel_obj = obj;
ecore_timer_del(sd->spin);
sd->spin = ecore_timer_add(sd->interval, _on_clock_val_down, data);
_on_clock_val_down(data);
}
static void
_on_clock_val_change_stop(void *data,
Evas_Object *obj EINA_UNUSED,
const char *emission EINA_UNUSED,
const char *source EINA_UNUSED)
{
ELM_CLOCK_DATA_GET(data, sd);
ELM_SAFE_FREE(sd->spin, ecore_timer_del);
sd->sel_obj = NULL;
}
static void
_access_activate_cb(void *data,
Evas_Object *part_obj,
Elm_Object_Item *item EINA_UNUSED)
{
Evas_Object *digit, *inc_btn;
ELM_CLOCK_DATA_GET(data, sd);
digit = evas_object_smart_parent_get(part_obj);
if (!digit) return;
edje_object_freeze(digit);
inc_btn = (Evas_Object *)edje_object_part_object_get(digit, "access.t");
edje_object_thaw(digit);
if (part_obj != inc_btn)
_on_clock_val_down_start(data, digit, NULL, NULL);
else
_on_clock_val_up_start(data, digit, NULL, NULL);
_on_clock_val_change_stop(sd, NULL, NULL, NULL);
}
static void
_access_time_register(Evas_Object *obj, Eina_Bool is_access)
{
Evas_Object *ao, *po;
ELM_CLOCK_DATA_GET(obj, sd);
if (!sd->edit) return;
/* hour, min, sec edit button */
int i;
for (i = 0; i < 6; i++)
{
if (is_access && (sd->digedit & (1 << i)))
{
char *digit = NULL;
switch (1 << i)
{
case ELM_CLOCK_EDIT_HOUR_DECIMAL:
digit = "hour decimal";
break;
case ELM_CLOCK_EDIT_HOUR_UNIT:
digit = "hour unit";
break;
case ELM_CLOCK_EDIT_MIN_DECIMAL:
digit = "minute decimal";
break;
case ELM_CLOCK_EDIT_MIN_UNIT:
digit = "minute unit";
break;
case ELM_CLOCK_EDIT_SEC_DECIMAL:
digit = "second decimal";
break;
case ELM_CLOCK_EDIT_SEC_UNIT:
digit = "second unit";
break;
}
Eina_Strbuf *strbuf;
strbuf = eina_strbuf_new();
/* increment button */
ao = _elm_access_edje_object_part_object_register
(obj, sd->digit[i], "access.t");
eina_strbuf_append_printf(strbuf,
"clock increment button for %s", digit);
_elm_access_text_set(_elm_access_info_get(ao),
ELM_ACCESS_TYPE, eina_strbuf_string_get(strbuf));
_elm_access_activate_callback_set
(_elm_access_info_get(ao), _access_activate_cb, obj);
/* decrement button */
ao = _elm_access_edje_object_part_object_register
(obj, sd->digit[i], "access.b");
eina_strbuf_replace(strbuf, "increment", "decrement", 1);
_elm_access_text_set(_elm_access_info_get(ao),
ELM_ACCESS_TYPE, eina_strbuf_string_get(strbuf));
_elm_access_activate_callback_set
(_elm_access_info_get(ao), _access_activate_cb, obj);
eina_strbuf_free(strbuf);
edje_object_signal_emit
(sd->digit[i], "elm,state,access,edit,on", "elm");
}
else if (!is_access && (sd->digedit & (1 << i)))
{
_elm_access_edje_object_part_object_unregister
(obj, sd->digit[i], "access.t");
_elm_access_edje_object_part_object_unregister
(obj, sd->digit[i], "access.b");
edje_object_signal_emit
(sd->digit[i], "elm,state,access,edit,off", "elm");
}
/* no need to propagate mouse event with acess */
edje_object_freeze(sd->digit[i]);
po = (Evas_Object *)edje_object_part_object_get
(sd->digit[i], "access.t");
evas_object_propagate_events_set(po, !is_access);
po = (Evas_Object *)edje_object_part_object_get
(sd->digit[i], "access.b");
evas_object_propagate_events_set(po, !is_access);
edje_object_thaw(sd->digit[i]);
}
/* am, pm edit button */
if (is_access && sd->am_pm)
{
/* increment button */
ao = _elm_access_edje_object_part_object_register
(obj, sd->am_pm_obj, "access.t");
_elm_access_text_set(_elm_access_info_get(ao),
ELM_ACCESS_TYPE, E_("clock increment button for am,pm"));
_elm_access_activate_callback_set
(_elm_access_info_get(ao), _access_activate_cb, obj);
/* decrement button */
ao = _elm_access_edje_object_part_object_register
(obj, sd->am_pm_obj, "access.b");
_elm_access_text_set(_elm_access_info_get(ao),
ELM_ACCESS_TYPE, E_("clock decrement button for am,pm"));
_elm_access_activate_callback_set
(_elm_access_info_get(ao), _access_activate_cb, obj);
edje_object_signal_emit
(sd->am_pm_obj, "elm,state,access,edit,on", "elm");
}
else if (!is_access && sd->am_pm)
{
_elm_access_edje_object_part_object_register
(obj, sd->am_pm_obj, "access.t");
_elm_access_edje_object_part_object_register
(obj, sd->am_pm_obj, "access.b");
edje_object_signal_emit
(sd->am_pm_obj, "elm,state,access,edit,off", "elm");
}
/* no need to propagate mouse event with access */
edje_object_freeze(sd->am_pm_obj);
po = (Evas_Object *)edje_object_part_object_get
(sd->am_pm_obj, "access.t");
evas_object_propagate_events_set(po, !is_access);
po = (Evas_Object *)edje_object_part_object_get
(sd->am_pm_obj, "access.b");
evas_object_propagate_events_set(po, !is_access);
edje_object_thaw(sd->am_pm_obj);
}
static Evas_Object*
_focus_part_get(Evas_Object *part, const char *part_name)
{
Evas_Object *po, *adapter;
edje_object_freeze(part);
po = (Evas_Object *)edje_object_part_object_get
(part, part_name);
edje_object_thaw(part);
if (_elm_config->access_mode == ELM_ACCESS_MODE_ON)
po = evas_object_data_get(po, "_part_access_obj");
adapter = evas_object_data_get(po, "_focus_adapter_object");
if (!adapter)
{
adapter = efl_add(EFL_UI_FOCUS_COMPOSITION_ADAPTER_CLASS, po);
efl_ui_focus_composition_adapter_canvas_object_set(adapter, part);
evas_object_data_set(po, "_focus_adapter_object", adapter);
}
return adapter;
}
static void
_flush_clock_composite_elements(Evas_Object *obj, Elm_Clock_Data *sd)
{
Eina_List *items = NULL;
int i;
if (sd->edit)
{
for (i = 0; i < 6; i++)
{
if ((!sd->seconds) && (i >= 4)) break;
if (sd->digedit & (1 << i))
{
items = eina_list_append(items, _focus_part_get(sd->digit[i], "access.t"));
items = eina_list_append(items, _focus_part_get(sd->digit[i], "access.b"));
}
}
if (sd->am_pm)
{
items = eina_list_append(items, _focus_part_get(sd->am_pm_obj, "access.t"));
items = eina_list_append(items, _focus_part_get(sd->am_pm_obj, "access.b"));
}
}
efl_ui_focus_composition_elements_set(obj, items);
}
static void
_time_update(Evas_Object *obj, Eina_Bool theme_update)
{
ELM_CLOCK_DATA_GET(obj, sd);
ELM_WIDGET_DATA_GET_OR_RETURN(obj, wd);
Edje_Message_Int msg;
const char *style = elm_widget_style_get(obj);
if ((sd->cur.seconds != sd->seconds) || (sd->cur.am_pm != sd->am_pm) ||
(sd->cur.edit != sd->edit) || (sd->cur.digedit != sd->digedit) || theme_update)
{
int i;
Evas_Coord mw, mh;
for (i = 0; i < 6; i++)
ELM_SAFE_FREE(sd->digit[i], evas_object_del);
ELM_SAFE_FREE(sd->am_pm_obj, evas_object_del);
if ((sd->seconds) && (sd->am_pm))
{
if (!elm_layout_theme_set(obj, "clock", "base-all", style))
CRI("Failed to set layout!");
}
else if (sd->seconds)
{
if (!elm_layout_theme_set(obj, "clock", "base-seconds", style))
CRI("Failed to set layout!");
}
else if (sd->am_pm)
{
if (!elm_layout_theme_set(obj, "clock", "base-am_pm", style))
CRI("Failed to set layout!");
}
else
{
if (!elm_layout_theme_set(obj, "clock", "base", style))
CRI("Failed to set layout!");
}
edje_object_scale_set
(wd->resize_obj, efl_gfx_entity_scale_get(obj) *
elm_config_scale_get());
for (i = 0; i < 6; i++)
{
char buf[16];
if ((!sd->seconds) && (i >= 4)) break;
sd->digit[i] = edje_object_add
(evas_object_evas_get(wd->resize_obj));
elm_widget_theme_object_set
(obj, sd->digit[i], "clock", "flipdigit", style);
edje_object_scale_set
(sd->digit[i], efl_gfx_entity_scale_get(obj) *
elm_config_scale_get());
if ((sd->edit) && (sd->digedit & (1 << i)))
edje_object_signal_emit
(sd->digit[i], "elm,state,edit,on", "elm");
edje_object_signal_callback_add
(sd->digit[i], "elm,action,up,start", "*",
_on_clock_val_up_start, obj);
edje_object_signal_callback_add
(sd->digit[i], "elm,action,up,stop", "*",
_on_clock_val_change_stop, obj);
edje_object_signal_callback_add
(sd->digit[i], "elm,action,down,start", "*",
_on_clock_val_down_start, obj);
edje_object_signal_callback_add
(sd->digit[i], "elm,action,down,stop", "*",
_on_clock_val_change_stop, obj);
mw = mh = -1;
elm_coords_finger_size_adjust(1, &mw, 2, &mh);
edje_object_size_min_restricted_calc
(sd->digit[i], &mw, &mh, mw, mh);
evas_object_size_hint_min_set(sd->digit[i], mw, mh);
snprintf(buf, sizeof(buf), "elm.d%i", i);
if (!elm_layout_content_set(obj, buf, sd->digit[i]))
{
// Previous versions of the theme did not have the namespace
snprintf(buf, sizeof(buf), "d%i", i);
elm_layout_content_set(obj, buf, sd->digit[i]);
}
evas_object_show(sd->digit[i]);
}
if (sd->am_pm)
{
sd->am_pm_obj =
edje_object_add(evas_object_evas_get(wd->resize_obj));
elm_widget_theme_object_set
(obj, sd->am_pm_obj, "clock", "flipampm", style);
edje_object_scale_set(sd->am_pm_obj, efl_gfx_entity_scale_get(obj) *
_elm_config->scale);
if (sd->edit)
edje_object_signal_emit
(sd->am_pm_obj, "elm,state,edit,on", "elm");
edje_object_signal_callback_add
(sd->am_pm_obj, "elm,action,up,start", "*",
_on_clock_val_up_start, obj);
edje_object_signal_callback_add
(sd->am_pm_obj, "elm,action,up,stop", "*",
_on_clock_val_change_stop, obj);
edje_object_signal_callback_add
(sd->am_pm_obj, "elm,action,down,start", "*",
_on_clock_val_down_start, obj);
edje_object_signal_callback_add
(sd->am_pm_obj, "elm,action,down,stop", "*",
_on_clock_val_change_stop, obj);
mw = mh = -1;
elm_coords_finger_size_adjust(1, &mw, 2, &mh);
edje_object_size_min_restricted_calc
(sd->am_pm_obj, &mw, &mh, mw, mh);
evas_object_size_hint_min_set(sd->am_pm_obj, mw, mh);
if (!elm_layout_content_set(obj, "elm.ampm", sd->am_pm_obj))
elm_layout_content_set(obj, "ampm", sd->am_pm_obj);
evas_object_show(sd->am_pm_obj);
}
/* access */
if (_elm_config->access_mode == ELM_ACCESS_MODE_ON)
_access_time_register(obj, EINA_TRUE);
edje_object_size_min_calc(wd->resize_obj, &mw, &mh);
evas_object_size_hint_min_set(obj, mw, mh);
sd->cur.hrs = 0;
sd->cur.min = 0;
sd->cur.sec = 0;
sd->cur.ampm = -1;
sd->cur.seconds = sd->seconds;
sd->cur.am_pm = sd->am_pm;
sd->cur.edit = sd->edit;
sd->cur.digedit = sd->digedit;
_flush_clock_composite_elements(obj, sd);
}
if (sd->hrs != sd->cur.hrs)
{
int hrs;
int d1, d2, dc1, dc2;
hrs = sd->hrs;
if (sd->am_pm)
{
if (hrs > 12) hrs -= 12;
else if (!hrs) hrs = 12;
}
d1 = hrs / 10;
d2 = hrs % 10;
dc1 = sd->cur.hrs / 10;
dc2 = sd->cur.hrs % 10;
if (d1 != dc1)
{
msg.val = d1;
edje_object_message_send(sd->digit[0], EDJE_MESSAGE_INT, 1, &msg);
}
if (d2 != dc2)
{
msg.val = d2;
edje_object_message_send(sd->digit[1], EDJE_MESSAGE_INT, 1, &msg);
}
sd->cur.hrs = hrs;
}
if (sd->min != sd->cur.min)
{
int d1, d2, dc1, dc2;
d1 = sd->min / 10;
d2 = sd->min % 10;
dc1 = sd->cur.min / 10;
dc2 = sd->cur.min % 10;
if (d1 != dc1)
{
msg.val = d1;
edje_object_message_send(sd->digit[2], EDJE_MESSAGE_INT, 1, &msg);
}
if (d2 != dc2)
{
msg.val = d2;
edje_object_message_send(sd->digit[3], EDJE_MESSAGE_INT, 1, &msg);
}
sd->cur.min = sd->min;
}
if (sd->seconds)
{
if (sd->sec != sd->cur.sec)
{
int d1, d2, dc1, dc2;
d1 = sd->sec / 10;
d2 = sd->sec % 10;
dc1 = sd->cur.sec / 10;
dc2 = sd->cur.sec % 10;
if (d1 != dc1)
{
msg.val = d1;
edje_object_message_send
(sd->digit[4], EDJE_MESSAGE_INT, 1, &msg);
}
if (d2 != dc2)
{
msg.val = d2;
edje_object_message_send
(sd->digit[5], EDJE_MESSAGE_INT, 1, &msg);
}
sd->cur.sec = sd->sec;
}
}
else
sd->cur.sec = -1;
if (sd->am_pm)
{
int ampm = 0;
if (sd->hrs >= 12) ampm = 1;
if (ampm != sd->cur.ampm)
{
msg.val = ampm;
edje_object_message_send
(sd->am_pm_obj, EDJE_MESSAGE_INT, 1, &msg);
sd->cur.ampm = ampm;
}
}
else
sd->cur.ampm = -1;
}
EOLIAN static Eina_Error
_elm_clock_efl_ui_widget_theme_apply(Eo *obj, Elm_Clock_Data *sd EINA_UNUSED)
{
Eina_Error int_ret = EFL_UI_THEME_APPLY_ERROR_GENERIC;
int_ret = efl_ui_widget_theme_apply(efl_super(obj, MY_CLASS));
if (int_ret == EFL_UI_THEME_APPLY_ERROR_GENERIC) return int_ret;
_time_update(obj, EINA_TRUE);
return int_ret;
}
static Eina_Bool
_ticker(void *data)
{
ELM_CLOCK_DATA_GET(data, sd);
double t;
struct timeval timev;
struct tm *tm;
time_t tt;
gettimeofday(&timev, NULL);
t = ((double)(1000000 - timev.tv_usec)) / 1000000.0;
sd->ticker = ecore_timer_add(t, _ticker, data);
if (!sd->edit)
{
tt = (time_t)(timev.tv_sec) + sd->timediff;
tzset();
tm = localtime(&tt);
if (tm)
{
sd->hrs = tm->tm_hour;
sd->min = tm->tm_min;
sd->sec = tm->tm_sec;
_time_update(data, EINA_FALSE);
}
}
return ECORE_CALLBACK_CANCEL;
}
static char *
_access_info_cb(void *data EINA_UNUSED, Evas_Object *obj)
{
int hrs;
char *ret;
Eina_Strbuf *buf;
ELM_CLOCK_DATA_GET(obj, sd);
buf = eina_strbuf_new();
hrs = sd->hrs;
if (sd->am_pm)
{
char *ampm = NULL;
if (hrs >= 12)
{
if (hrs > 12) hrs -= 12;
ampm = "PM";
}
else ampm = "AM";
eina_strbuf_append_printf(buf, "%d, %d, %s", hrs, sd->min, ampm);
}
else
{
eina_strbuf_append_printf(buf, "%d, %d", hrs, sd->min);
}
ret = eina_strbuf_string_steal(buf);
eina_strbuf_free(buf);
return ret;
}
static char *
_access_state_cb(void *data EINA_UNUSED, Evas_Object *obj)
{
ELM_CLOCK_DATA_GET(obj, sd);
if (sd->edit)
return strdup(E_("State: Editable"));
return NULL;
}
EOLIAN static void
_elm_clock_efl_canvas_group_group_add(Eo *obj, Elm_Clock_Data *priv)
{
ELM_WIDGET_DATA_GET_OR_RETURN(obj, wd);
efl_canvas_group_add(efl_super(obj, MY_CLASS));
priv->cur.ampm = -1;
priv->cur.seconds = EINA_TRUE;
priv->cur.am_pm = EINA_TRUE;
priv->cur.edit = EINA_TRUE;
priv->cur.digedit = ELM_CLOCK_EDIT_DEFAULT;
priv->first_interval = DEFAULT_FIRST_INTERVAL;
elm_widget_can_focus_set(obj, EINA_TRUE);
_time_update(obj, EINA_FALSE);
_ticker(obj);
/* access */
if (_elm_config->access_mode != ELM_ACCESS_MODE_OFF)
{
evas_object_propagate_events_set(obj, EINA_FALSE);
edje_object_signal_emit(wd->resize_obj,
"elm,state,access,on", "elm");
}
_elm_access_object_register(obj, wd->resize_obj);
_elm_access_text_set
(_elm_access_info_get(obj), ELM_ACCESS_TYPE, E_("Clock"));
_elm_access_callback_set
(_elm_access_info_get(obj), ELM_ACCESS_INFO, _access_info_cb, NULL);
evas_object_propagate_events_set(obj, EINA_FALSE);
_elm_access_callback_set
(_elm_access_info_get(obj), ELM_ACCESS_STATE, _access_state_cb, NULL);
}
EOLIAN static void
_elm_clock_efl_canvas_group_group_del(Eo *obj, Elm_Clock_Data *sd)
{
ecore_timer_del(sd->ticker);
ecore_timer_del(sd->spin);
/* NB: digits are killed for being sub objects, automatically */
efl_canvas_group_del(efl_super(obj, MY_CLASS));
}
static Eina_Bool _elm_clock_smart_focus_next_enable = EINA_FALSE;
static void
_access_obj_process(Evas_Object *obj, Eina_Bool is_access)
{
ELM_WIDGET_DATA_GET_OR_RETURN(obj, wd);
/* clock object */
evas_object_propagate_events_set(obj, !is_access);
if (is_access)
edje_object_signal_emit(wd->resize_obj,
"elm,state,access,on", "elm");
else
edje_object_signal_emit(wd->resize_obj,
"elm,state,access,off", "elm");
/* clock time objects */
_access_time_register(obj, is_access);
}
EOLIAN static void
_elm_clock_efl_ui_widget_on_access_update(Eo *obj EINA_UNUSED, Elm_Clock_Data *_pd EINA_UNUSED, Eina_Bool acs)
{
_elm_clock_smart_focus_next_enable = acs;
_access_obj_process(obj, _elm_clock_smart_focus_next_enable);
}
EAPI Evas_Object *
elm_clock_add(Evas_Object *parent)
{
EINA_SAFETY_ON_NULL_RETURN_VAL(parent, NULL);
return elm_legacy_add(MY_CLASS, parent);
}
EOLIAN static Eo *
_elm_clock_efl_object_constructor(Eo *obj, Elm_Clock_Data *_pd EINA_UNUSED)
{
obj = efl_constructor(efl_super(obj, MY_CLASS));
efl_canvas_object_type_set(obj, MY_CLASS_NAME_LEGACY);
evas_object_smart_callbacks_descriptions_set(obj, _smart_callbacks);
efl_access_object_role_set(obj, EFL_ACCESS_ROLE_TEXT);
legacy_child_focus_handle(obj);
return obj;
}
static void
_timediff_set(Elm_Clock_Data *sd)
{
struct timeval timev;
struct tm *tm;
time_t tt;
gettimeofday(&timev, NULL);
tt = (time_t)(timev.tv_sec);
tzset();
tm = localtime(&tt);
if (tm)
{
sd->timediff = (((sd->hrs - tm->tm_hour) * 60 +
sd->min - tm->tm_min) * 60) + sd->sec - tm->tm_sec;
}
else
{
ERR("Failed to get local time!");
sd->timediff = 0;
}
}
EOLIAN static void
_elm_clock_time_set(Eo *obj, Elm_Clock_Data *sd, int hrs, int min, int sec)
{
sd->hrs = hrs;
sd->min = min;
sd->sec = sec;
_timediff_set(sd);
_time_update(obj, EINA_FALSE);
}
EOLIAN static void
_elm_clock_time_get(const Eo *obj EINA_UNUSED, Elm_Clock_Data *sd, int *hrs, int *min, int *sec)
{
if (hrs) *hrs = sd->hrs;
if (min) *min = sd->min;
if (sec) *sec = sd->sec;
}
EOLIAN static void
_elm_clock_edit_set(Eo *obj, Elm_Clock_Data *sd, Eina_Bool edit)
{
sd->edit = edit;
if (!edit)
_timediff_set(sd);
if ((edit) && (sd->digedit == ELM_CLOCK_EDIT_DEFAULT))
elm_clock_edit_mode_set(obj, ELM_CLOCK_EDIT_ALL);
else
_time_update(obj, EINA_FALSE);
}
EOLIAN static Eina_Bool
_elm_clock_edit_get(const Eo *obj EINA_UNUSED, Elm_Clock_Data *sd)
{
return sd->edit;
}
EOLIAN static void
_elm_clock_edit_mode_set(Eo *obj, Elm_Clock_Data *sd, Elm_Clock_Edit_Mode digedit)
{
sd->digedit = digedit;
if (digedit == ELM_CLOCK_EDIT_DEFAULT)
elm_clock_edit_set(obj, EINA_FALSE);
else
_time_update(obj, EINA_FALSE);
}
EOLIAN static Elm_Clock_Edit_Mode
_elm_clock_edit_mode_get(const Eo *obj EINA_UNUSED, Elm_Clock_Data *sd)
{
return sd->digedit;
}
EOLIAN static void
_elm_clock_show_am_pm_set(Eo *obj, Elm_Clock_Data *sd, Eina_Bool am_pm)
{
sd->am_pm = !!am_pm;
_time_update(obj, EINA_FALSE);
}
EOLIAN static Eina_Bool
_elm_clock_show_am_pm_get(const Eo *obj EINA_UNUSED, Elm_Clock_Data *sd)
{
return sd->am_pm;
}
EOLIAN static void
_elm_clock_show_seconds_set(Eo *obj, Elm_Clock_Data *sd, Eina_Bool seconds)
{
sd->seconds = !!seconds;
_time_update(obj, EINA_FALSE);
}
EOLIAN static Eina_Bool
_elm_clock_show_seconds_get(const Eo *obj EINA_UNUSED, Elm_Clock_Data *sd)
{
return sd->seconds;
}
EOLIAN static void
_elm_clock_first_interval_set(Eo *obj EINA_UNUSED, Elm_Clock_Data *sd, double interval)
{
sd->first_interval = interval;
}
EOLIAN static double
_elm_clock_first_interval_get(const Eo *obj EINA_UNUSED, Elm_Clock_Data *sd)
{
return sd->first_interval;
}
EOLIAN static void
_elm_clock_pause_set(Eo *obj EINA_UNUSED, Elm_Clock_Data *sd, Eina_Bool paused)
{
paused = !!paused;
if (sd->paused == paused)
return;
sd->paused = paused;
if (paused)
ecore_timer_freeze(sd->ticker);
else
{
_timediff_set(sd);
ecore_timer_thaw(sd->ticker);
}
}
EOLIAN static Eina_Bool
_elm_clock_pause_get(const Eo *obj EINA_UNUSED, Elm_Clock_Data *sd)
{
return sd->paused;
}
static void
_elm_clock_class_constructor(Efl_Class *klass)
{
evas_smart_legacy_type_register(MY_CLASS_NAME_LEGACY, klass);
if (_elm_config->access_mode != ELM_ACCESS_MODE_OFF)
_elm_clock_smart_focus_next_enable = EINA_TRUE;
}
/* Internal EO APIs and hidden overrides */
#define ELM_CLOCK_EXTRA_OPS \
EFL_CANVAS_GROUP_ADD_DEL_OPS(elm_clock)
#include "elm_clock_eo.c"