forked from enlightenment/efl
961 lines
26 KiB
C
961 lines
26 KiB
C
#ifdef HAVE_CONFIG_H
|
|
# include "elementary_config.h"
|
|
#endif
|
|
|
|
#define ELM_INTERFACE_ATSPI_ACCESSIBLE_PROTECTED
|
|
|
|
#include <Elementary.h>
|
|
#include "elm_priv.h"
|
|
#include "elm_widget_clock.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;
|
|
|
|
inc_btn = (Evas_Object *)edje_object_part_object_get(digit, "access.t");
|
|
|
|
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 */
|
|
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);
|
|
|
|
}
|
|
|
|
/* 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 */
|
|
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);
|
|
|
|
}
|
|
|
|
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, elm_widget_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], elm_widget_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), "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, elm_widget_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);
|
|
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;
|
|
}
|
|
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 Elm_Theme_Apply
|
|
_elm_clock_elm_widget_theme_apply(Eo *obj, Elm_Clock_Data *sd EINA_UNUSED)
|
|
{
|
|
Elm_Theme_Apply int_ret = ELM_THEME_APPLY_FAILED;
|
|
|
|
int_ret = elm_obj_widget_theme_apply(efl_super(obj, MY_CLASS));
|
|
if (!int_ret) return ELM_THEME_APPLY_FAILED;
|
|
|
|
_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));
|
|
elm_widget_sub_object_parent_add(obj);
|
|
|
|
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;
|
|
|
|
EOLIAN static Eina_Bool
|
|
_elm_clock_elm_widget_focus_next_manager_is(Eo *obj EINA_UNUSED, Elm_Clock_Data *_pd EINA_UNUSED)
|
|
{
|
|
return _elm_clock_smart_focus_next_enable;
|
|
}
|
|
|
|
EOLIAN static Eina_Bool
|
|
_elm_clock_elm_widget_focus_next(Eo *obj, Elm_Clock_Data *sd, Elm_Focus_Direction dir, Evas_Object **next, Elm_Object_Item **next_item)
|
|
{
|
|
Evas_Object *ao, *po;
|
|
Eina_List *items = NULL;
|
|
|
|
|
|
if (!sd->edit)
|
|
{
|
|
*next = (Evas_Object *)obj;
|
|
return !elm_widget_highlight_get(obj);
|
|
}
|
|
else if (!elm_widget_highlight_get(obj))
|
|
{
|
|
*next = (Evas_Object *)obj;
|
|
return EINA_TRUE;
|
|
}
|
|
|
|
int i;
|
|
for (i = 0; i < 6; i++)
|
|
{
|
|
if ((!sd->seconds) && (i >= 4)) break;
|
|
if (sd->digedit & (1 << i))
|
|
{
|
|
po = (Evas_Object *)edje_object_part_object_get
|
|
(sd->digit[i], "access.t");
|
|
ao = evas_object_data_get(po, "_part_access_obj");
|
|
items = eina_list_append(items, ao);
|
|
|
|
po = (Evas_Object *)edje_object_part_object_get
|
|
(sd->digit[i], "access.b");
|
|
ao = evas_object_data_get(po, "_part_access_obj");
|
|
items = eina_list_append(items, ao);
|
|
}
|
|
}
|
|
|
|
if (sd->am_pm)
|
|
{
|
|
po = (Evas_Object *)edje_object_part_object_get
|
|
(sd->am_pm_obj, "access.t");
|
|
ao = evas_object_data_get(po, "_part_access_obj");
|
|
items = eina_list_append(items, ao);
|
|
|
|
po = (Evas_Object *)edje_object_part_object_get
|
|
(sd->am_pm_obj, "access.b");
|
|
ao = evas_object_data_get(po, "_part_access_obj");
|
|
items = eina_list_append(items, ao);
|
|
}
|
|
|
|
return elm_widget_focus_list_next_get
|
|
(obj, items, eina_list_data_get, dir, next, next_item);
|
|
}
|
|
|
|
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_elm_widget_access(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);
|
|
Evas_Object *obj = efl_add(MY_CLASS, parent);
|
|
return obj;
|
|
}
|
|
|
|
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);
|
|
elm_interface_atspi_accessible_role_set(obj, ELM_ATSPI_ROLE_TEXT);
|
|
|
|
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(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(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(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(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(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(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(Eo *obj EINA_UNUSED, Elm_Clock_Data *sd)
|
|
{
|
|
return sd->paused;
|
|
}
|
|
|
|
EOLIAN static Eina_Bool
|
|
_elm_clock_elm_widget_focus_direction_manager_is(Eo *obj EINA_UNUSED, Elm_Clock_Data *_pd EINA_UNUSED)
|
|
{
|
|
return EINA_FALSE;
|
|
}
|
|
|
|
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;
|
|
}
|
|
|
|
#include "elm_clock.eo.c"
|