efl/legacy/elementary/src/lib/elm_clock.c

713 lines
19 KiB
C
Raw Normal View History

#include <Elementary.h>
#include "elm_priv.h"
#include "elm_widget_layout.h"
static const char CLOCK_SMART_NAME[] = "elm_clock";
#define DEFAULT_FIRST_INTERVAL 0.85
typedef struct _Elm_Clock_Smart_Data Elm_Clock_Smart_Data;
struct _Elm_Clock_Smart_Data
{
Elm_Layout_Smart_Data base;
double interval, first_interval;
Elm_Clock_Edit_Mode digedit;
int hrs, min, sec, timediff;
Evas_Object *digit[6];
Evas_Object *am_pm_obj;
Evas_Object *sel_obj;
Ecore_Timer *ticker, *spin;
struct
{
int hrs, min, sec;
char ampm;
Elm_Clock_Edit_Mode digedit;
Eina_Bool seconds : 1;
Eina_Bool am_pm : 1;
Eina_Bool edit : 1;
} cur;
Eina_Bool seconds : 1;
Eina_Bool am_pm : 1;
Eina_Bool edit : 1;
};
#define ELM_CLOCK_DATA_GET(o, sd) \
Elm_Clock_Smart_Data * sd = evas_object_smart_data_get(o)
#define ELM_CLOCK_DATA_GET_OR_RETURN(o, ptr) \
ELM_CLOCK_DATA_GET(o, ptr); \
if (!ptr) \
{ \
CRITICAL("No widget data for object %p (%s)", \
o, evas_object_type_get(o)); \
return; \
}
#define ELM_CLOCK_DATA_GET_OR_RETURN_VAL(o, ptr, val) \
ELM_CLOCK_DATA_GET(o, ptr); \
if (!ptr) \
{ \
CRITICAL("No widget data for object %p (%s)", \
o, evas_object_type_get(o)); \
return val; \
}
#define ELM_CLOCK_CHECK(obj) \
if (!obj || !elm_widget_type_check((obj), CLOCK_SMART_NAME, __func__)) \
return
static void _time_update(Evas_Object *obj);
static const char SIG_CHANGED[] = "changed";
static const Evas_Smart_Cb_Description _smart_callbacks[] = {
{SIG_CHANGED, ""},
{NULL, NULL}
};
/* Inheriting from elm_layout. Besides, we need no more than what is
* there */
EVAS_SMART_SUBCLASS_NEW
(CLOCK_SMART_NAME, _elm_clock, Elm_Layout_Smart_Class,
Elm_Layout_Smart_Class, elm_layout_smart_class_get, _smart_callbacks);
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 + 10;
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);
evas_object_smart_callback_call(data, SIG_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 - 10;
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);
evas_object_smart_callback_call(data, SIG_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 __UNUSED__,
const char *source __UNUSED__)
{
ELM_CLOCK_DATA_GET(data, sd);
sd->interval = sd->first_interval;
sd->sel_obj = obj;
if (sd->spin) 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 __UNUSED__,
const char *source __UNUSED__)
{
ELM_CLOCK_DATA_GET(data, sd);
sd->interval = sd->first_interval;
sd->sel_obj = obj;
if (sd->spin) 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 __UNUSED__,
const char *emission __UNUSED__,
const char *source __UNUSED__)
{
Elm_Clock_Smart_Data *sd = data;
if (sd->spin) ecore_timer_del(sd->spin);
sd->spin = NULL;
sd->sel_obj = NULL;
}
static void
_time_update(Evas_Object *obj)
{
ELM_CLOCK_DATA_GET(obj, sd);
Edje_Message_Int msg;
int ampm = 0;
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))
{
int i;
Evas_Coord mw, mh;
for (i = 0; i < 6; i++)
{
if (sd->digit[i])
{
evas_object_del(sd->digit[i]);
sd->digit[i] = NULL;
}
}
if (sd->am_pm_obj)
{
evas_object_del(sd->am_pm_obj);
sd->am_pm_obj = NULL;
}
if ((sd->seconds) && (sd->am_pm))
elm_layout_theme_set(obj, "clock", "base-all", style);
else if (sd->seconds)
elm_layout_theme_set(obj, "clock", "base-seconds", style);
else if (sd->am_pm)
elm_layout_theme_set(obj, "clock", "base-am_pm", style);
else
elm_layout_theme_set(obj, "clock", "base", style);
edje_object_scale_set
(ELM_WIDGET_DATA(sd)->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(ELM_WIDGET_DATA(sd)->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, sd);
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, sd);
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);
elm_coords_finger_size_adjust(1, &mw, 2, &mh);
edje_extern_object_min_size_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(ELM_WIDGET_DATA(sd)->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, sd);
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, sd);
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);
elm_coords_finger_size_adjust(1, &mw, 2, &mh);
edje_extern_object_min_size_set(sd->am_pm_obj, mw, mh);
elm_layout_content_set(obj, "ampm", sd->am_pm_obj);
evas_object_show(sd->am_pm_obj);
}
edje_object_size_min_calc(ELM_WIDGET_DATA(sd)->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)
{
if (hrs > 12) hrs -= 12;
ampm = 1;
}
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)
{
if (sd->hrs >= 12) ampm = 1;
if (ampm != sd->cur.ampm)
{
if (sd->cur.ampm != 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;
}
static Eina_Bool
_elm_clock_smart_theme(Evas_Object *obj)
{
if (!ELM_WIDGET_CLASS(_elm_clock_parent_sc)->theme(obj)) return EINA_FALSE;
_time_update(obj);
return EINA_TRUE;
}
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);
}
}
return ECORE_CALLBACK_CANCEL;
}
static void
_elm_clock_smart_add(Evas_Object *obj)
{
EVAS_SMART_DATA_ALLOC(obj, Elm_Clock_Smart_Data);
ELM_WIDGET_CLASS(_elm_clock_parent_sc)->base.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;
priv->timediff = 0;
elm_widget_can_focus_set(obj, EINA_TRUE);
_time_update(obj);
_ticker(obj);
}
static void
_elm_clock_smart_del(Evas_Object *obj)
{
ELM_CLOCK_DATA_GET(obj, sd);
if (sd->ticker) ecore_timer_del(sd->ticker);
if (sd->spin) ecore_timer_del(sd->spin);
/* NB: digits are killed for being sub objects, automatically */
ELM_WIDGET_CLASS(_elm_clock_parent_sc)->base.del(obj);
}
static void
_elm_clock_smart_set_user(Elm_Layout_Smart_Class *sc)
{
ELM_WIDGET_CLASS(sc)->base.add = _elm_clock_smart_add;
ELM_WIDGET_CLASS(sc)->base.del = _elm_clock_smart_del;
ELM_WIDGET_CLASS(sc)->theme = _elm_clock_smart_theme;
/* not a 'focus chain manager' */
ELM_WIDGET_CLASS(sc)->focus_next = NULL;
ELM_WIDGET_CLASS(sc)->focus_direction = NULL;
}
EAPI Evas_Object *
elm_clock_add(Evas_Object *parent)
{
Evas_Object *obj;
EINA_SAFETY_ON_NULL_RETURN_VAL(parent, NULL);
obj = elm_widget_add(_elm_clock_smart_class_new(), parent);
if (!obj) return NULL;
if (!elm_widget_sub_object_add(parent, obj))
ERR("could not add %p as sub object of %p", obj, parent);
return obj;
}
static void
_timediff_set(Elm_Clock_Smart_Data *sd)
{
struct timeval timev;
struct tm *tm;
time_t tt;
gettimeofday(&timev, NULL);
tt = (time_t)(timev.tv_sec);
tzset();
tm = localtime(&tt);
sd->timediff = (((sd->hrs - tm->tm_hour) * 60 +
sd->min - tm->tm_min) * 60) + sd->sec - tm->tm_sec;
}
EAPI void
elm_clock_time_set(Evas_Object *obj,
int hrs,
int min,
int sec)
{
ELM_CLOCK_CHECK(obj);
ELM_CLOCK_DATA_GET(obj, sd);
sd->hrs = hrs;
sd->min = min;
sd->sec = sec;
_timediff_set(sd);
_time_update(obj);
}
EAPI void
elm_clock_time_get(const Evas_Object *obj,
int *hrs,
int *min,
int *sec)
{
ELM_CLOCK_CHECK(obj);
ELM_CLOCK_DATA_GET(obj, sd);
if (hrs) *hrs = sd->hrs;
if (min) *min = sd->min;
if (sec) *sec = sd->sec;
}
EAPI void
elm_clock_edit_set(Evas_Object *obj,
Eina_Bool edit)
{
ELM_CLOCK_CHECK(obj);
ELM_CLOCK_DATA_GET(obj, sd);
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);
}
EAPI Eina_Bool
elm_clock_edit_get(const Evas_Object *obj)
{
ELM_CLOCK_CHECK(obj) EINA_FALSE;
ELM_CLOCK_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
return sd->edit;
}
EAPI void
elm_clock_edit_mode_set(Evas_Object *obj,
Elm_Clock_Edit_Mode digedit)
{
ELM_CLOCK_CHECK(obj);
ELM_CLOCK_DATA_GET(obj, sd);
sd->digedit = digedit;
if (digedit == ELM_CLOCK_EDIT_DEFAULT)
elm_clock_edit_set(obj, EINA_FALSE);
else
_time_update(obj);
}
EAPI Elm_Clock_Edit_Mode
elm_clock_edit_mode_get(const Evas_Object *obj)
{
ELM_CLOCK_CHECK(obj) 0;
ELM_CLOCK_DATA_GET_OR_RETURN_VAL(obj, sd, 0);
return sd->digedit;
}
EAPI void
elm_clock_show_am_pm_set(Evas_Object *obj,
Eina_Bool am_pm)
{
ELM_CLOCK_CHECK(obj);
ELM_CLOCK_DATA_GET(obj, sd);
sd->am_pm = !!am_pm;
_time_update(obj);
}
EAPI Eina_Bool
elm_clock_show_am_pm_get(const Evas_Object *obj)
{
ELM_CLOCK_CHECK(obj) EINA_FALSE;
ELM_CLOCK_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
return sd->am_pm;
}
EAPI void
elm_clock_show_seconds_set(Evas_Object *obj,
Eina_Bool seconds)
{
ELM_CLOCK_CHECK(obj);
ELM_CLOCK_DATA_GET(obj, sd);
sd->seconds = !!seconds;
_time_update(obj);
}
EAPI Eina_Bool
elm_clock_show_seconds_get(const Evas_Object *obj)
{
ELM_CLOCK_CHECK(obj) EINA_FALSE;
ELM_CLOCK_DATA_GET_OR_RETURN_VAL(obj, sd, EINA_FALSE);
return sd->seconds;
}
EAPI void
elm_clock_first_interval_set(Evas_Object *obj,
double interval)
{
ELM_CLOCK_CHECK(obj);
ELM_CLOCK_DATA_GET(obj, sd);
sd->first_interval = !!interval;
}
EAPI double
elm_clock_first_interval_get(const Evas_Object *obj)
{
ELM_CLOCK_CHECK(obj) 0.0;
ELM_CLOCK_DATA_GET_OR_RETURN_VAL(obj, sd, 0.0);
return sd->first_interval;
}