1 #include <Elementary.h>
5 typedef struct _Widget_Data Widget_Data;
6 typedef struct _Elm_Spinner_Special_Value Elm_Spinner_Special_Value;
10 Evas_Object *spinner, *ent;
12 double val, val_min, val_max, orig_val, step, base;
13 double drag_start_pos, spin_speed, interval, first_interval;
15 Ecore_Timer *delay, *spin;
16 Eina_List *special_values;
18 Eina_Bool entry_visible : 1;
19 Eina_Bool dragging : 1;
20 Eina_Bool editable : 1;
23 struct _Elm_Spinner_Special_Value
29 static const char *widtype = NULL;
30 static void _del_hook(Evas_Object *obj);
31 static void _disable_hook(Evas_Object *obj);
32 static void _write_label(Evas_Object *obj);
33 static void _sizing_eval(Evas_Object *obj);
34 //static void _changed_size_hints(void *data, Evas *e, Evas_Object *obj, void *event_info);
35 static Eina_Bool _value_set(Evas_Object *obj, double delta);
36 static void _on_focus_hook(void *data, Evas_Object *obj);
37 static Eina_Bool _event_hook(Evas_Object *obj, Evas_Object *src,
38 Evas_Callback_Type type, void *event_info);
40 static void _mirrored_set(Evas_Object *obj, Eina_Bool rtl);
42 static const char SIG_CHANGED[] = "changed";
43 static const char SIG_DELAY_CHANGED[] = "delay,changed";
45 static const Evas_Smart_Cb_Description _signals[] = {
47 {SIG_DELAY_CHANGED, ""},
52 _del_hook(Evas_Object *obj)
54 Elm_Spinner_Special_Value *sv;
55 Widget_Data *wd = elm_widget_data_get(obj);
57 if (wd->label) eina_stringshare_del(wd->label);
58 if (wd->delay) ecore_timer_del(wd->delay);
59 if (wd->spin) ecore_timer_del(wd->spin);
60 if (wd->special_values)
62 EINA_LIST_FREE(wd->special_values, sv)
64 eina_stringshare_del(sv->label);
72 _disable_hook(Evas_Object *obj)
74 Widget_Data *wd = elm_widget_data_get(obj);
76 if (elm_widget_disabled_get(obj))
77 edje_object_signal_emit(wd->spinner, "elm,state,disabled", "elm");
79 edje_object_signal_emit(wd->spinner, "elm,state,enabled", "elm");
83 _signal_emit_hook(Evas_Object *obj, const char *emission, const char *source)
85 Widget_Data *wd = elm_widget_data_get(obj);
87 edje_object_signal_emit(wd->spinner, emission, source);
91 _signal_callback_add_hook(Evas_Object *obj, const char *emission, const char *source, Edje_Signal_Cb func_cb, void *data)
93 Widget_Data *wd = elm_widget_data_get(obj);
95 edje_object_signal_callback_add(wd->spinner, emission,
96 source, func_cb, data);
100 _signal_callback_del_hook(Evas_Object *obj, const char *emission, const char *source, Edje_Signal_Cb func_cb, void *data)
102 Widget_Data *wd = elm_widget_data_get(obj);
103 edje_object_signal_callback_del_full(wd->spinner, emission, source,
108 _mirrored_set(Evas_Object *obj, Eina_Bool rtl)
110 Widget_Data *wd = elm_widget_data_get(obj);
112 edje_object_mirrored_set(wd->spinner, rtl);
116 _theme_hook(Evas_Object *obj)
118 Widget_Data *wd = elm_widget_data_get(obj);
120 _elm_widget_mirrored_reload(obj);
121 _mirrored_set(obj, elm_widget_mirrored_get(obj));
122 _elm_theme_object_set(obj, wd->spinner, "spinner", "base", elm_widget_style_get(obj));
123 edje_object_part_swallow(wd->spinner, "elm.swallow.entry", wd->ent);
125 if (elm_widget_focus_get(obj))
126 edje_object_signal_emit(wd->spinner, "elm,action,focus", "elm");
128 edje_object_signal_emit(wd->spinner, "elm,action,unfocus", "elm");
129 if (elm_widget_disabled_get(obj))
130 edje_object_signal_emit(wd->spinner, "elm,state,disabled", "elm");
131 edje_object_message_signal_process(wd->spinner);
132 edje_object_scale_set(wd->spinner, elm_widget_scale_get(obj) * _elm_config->scale);
137 _on_focus_hook(void *data __UNUSED__, Evas_Object *obj)
139 Widget_Data *wd = elm_widget_data_get(obj);
141 if (elm_widget_focus_get(obj))
143 edje_object_signal_emit(wd->spinner, "elm,action,focus", "elm");
144 evas_object_focus_set(wd->spinner, EINA_TRUE);
148 edje_object_signal_emit(wd->spinner, "elm,action,unfocus", "elm");
149 evas_object_focus_set(wd->spinner, EINA_FALSE);
154 _delay_change(void *data)
156 Widget_Data *wd = elm_widget_data_get(data);
157 if (!wd) return ECORE_CALLBACK_CANCEL;
159 evas_object_smart_callback_call(data, SIG_DELAY_CHANGED, NULL);
160 return ECORE_CALLBACK_CANCEL;
164 _entry_show(Widget_Data *wd)
166 char buf[32], fmt[32] = "%0.f";
168 /* try to construct just the format from given label
169 * completely ignoring pre/post words
173 const char *start = strchr(wd->label, '%');
180 start = strchr(start + 2, '%');
185 const char *itr, *end = NULL;
186 for (itr = start + 1; *itr != '\0'; itr++)
188 /* allowing '%d' is quite dangerous, remove it? */
189 if ((*itr == 'd') || (*itr == 'f'))
196 if ((end) && ((size_t)(end - start + 1) < sizeof(fmt)))
198 memcpy(fmt, start, end - start);
199 fmt[end - start] = '\0';
203 snprintf(buf, sizeof(buf), fmt, wd->val);
204 elm_object_text_set(wd->ent, buf);
208 _write_label(Evas_Object *obj)
211 Elm_Spinner_Special_Value *sv;
212 Widget_Data *wd = elm_widget_data_get(obj);
215 EINA_LIST_FOREACH(wd->special_values, l, sv)
217 if (sv->value == wd->val)
219 snprintf(buf, sizeof(buf), "%s", sv->label);
224 snprintf(buf, sizeof(buf), wd->label, wd->val);
226 snprintf(buf, sizeof(buf), "%.0f", wd->val);
229 edje_object_part_text_set(wd->spinner, "elm.text", buf);
230 if (wd->entry_visible) _entry_show(wd);
234 _value_set(Evas_Object *obj, double new_val)
236 Widget_Data *wd = elm_widget_data_get(obj);
238 if (!wd) return EINA_FALSE;
242 (double)((((int)(new_val - wd->base)) / wd->round) * wd->round);
246 while (new_val < wd->val_min)
247 new_val = wd->val_max + new_val + 1 - wd->val_min;
248 while (new_val > wd->val_max)
249 new_val = wd->val_min + new_val - wd->val_max - 1;
253 if (new_val < wd->val_min)
254 new_val = wd->val_min;
255 else if (new_val > wd->val_max)
256 new_val = wd->val_max;
259 if (new_val == wd->val) return EINA_FALSE;
262 evas_object_smart_callback_call(obj, SIG_CHANGED, NULL);
263 if (wd->delay) ecore_timer_del(wd->delay);
264 wd->delay = ecore_timer_add(0.2, _delay_change, obj);
270 _sizing_eval(Evas_Object *obj)
272 Widget_Data *wd = elm_widget_data_get(obj);
273 Evas_Coord minw = -1, minh = -1;
275 elm_coords_finger_size_adjust(1, &minw, 1, &minh);
276 edje_object_size_min_restricted_calc(wd->spinner, &minw, &minh, minw, minh);
277 elm_coords_finger_size_adjust(1, &minw, 1, &minh);
278 evas_object_size_hint_min_set(obj, minw, minh);
279 evas_object_size_hint_max_set(obj, -1, -1);
284 _changed_size_hints(void *data, Evas *e, Evas_Object *obj, void *event_info)
291 _val_set(Evas_Object *obj)
293 Widget_Data *wd = elm_widget_data_get(obj);
296 if (wd->val_max > wd->val_min)
297 pos = ((wd->val - wd->val_min) / (wd->val_max - wd->val_min));
298 if (pos < 0.0) pos = 0.0;
299 else if (pos > 1.0) pos = 1.0;
300 edje_object_part_drag_value_set(wd->spinner, "elm.dragable.slider",
305 _drag(void *data, Evas_Object *_obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
307 Evas_Object *obj = data;
308 Widget_Data *wd = elm_widget_data_get(obj);
309 double pos = 0.0, offset, delta;
311 if (wd->entry_visible) return;
312 edje_object_part_drag_value_get(wd->spinner, "elm.dragable.slider",
314 offset = wd->step * _elm_config->scale;
315 delta = (pos - wd->drag_start_pos) * offset;
316 /* If we are on rtl mode, change the delta to be negative on such changes */
317 if (elm_widget_mirrored_get(obj)) delta *= -1;
318 if (_value_set(data, wd->drag_start_pos + delta)) _write_label(data);
323 _drag_start(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
325 Widget_Data *wd = elm_widget_data_get(data);
328 edje_object_part_drag_value_get(wd->spinner, "elm.dragable.slider",
330 wd->drag_start_pos = pos;
334 _drag_stop(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
336 Widget_Data *wd = elm_widget_data_get(data);
338 wd->drag_start_pos = 0;
339 edje_object_part_drag_value_set(wd->spinner, "elm.dragable.slider", 0.0, 0.0);
343 _hide_entry(Evas_Object *obj)
345 Widget_Data *wd = elm_widget_data_get(obj);
347 edje_object_signal_emit(wd->spinner, "elm,state,inactive", "elm");
348 wd->entry_visible = 0;
352 _reset_value(Evas_Object *obj)
354 Widget_Data *wd = elm_widget_data_get(obj);
357 elm_spinner_value_set(obj, wd->orig_val);
361 _apply_entry_value(Evas_Object *obj)
363 Widget_Data *wd = elm_widget_data_get(obj);
370 str = elm_object_text_get(wd->ent);
372 val = strtod(str, &end);
373 if ((*end != '\0') && (!isspace(*end))) return;
374 elm_spinner_value_set(obj, val);
378 _toggle_entry(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
380 Widget_Data *wd = elm_widget_data_get(data);
387 if (elm_widget_disabled_get(data)) return;
388 if (!wd->editable) return;
389 if (wd->entry_visible) _apply_entry_value(data);
392 wd->orig_val = wd->val;
393 edje_object_signal_emit(wd->spinner, "elm,state,active", "elm");
395 elm_entry_select_all(wd->ent);
396 elm_widget_focus_set(wd->ent, 1);
397 wd->entry_visible = 1;
402 _spin_value(void *data)
404 Widget_Data *wd = elm_widget_data_get(data);
405 if (!wd) return ECORE_CALLBACK_CANCEL;
406 if (_value_set(data, wd->val + wd->spin_speed)) _write_label(data);
407 wd->interval = wd->interval / 1.05;
408 ecore_timer_interval_set(wd->spin, wd->interval);
409 return ECORE_CALLBACK_RENEW;
413 _val_inc_start(Evas_Object *obj)
415 Widget_Data *wd = elm_widget_data_get(obj);
417 wd->interval = wd->first_interval;
418 wd->spin_speed = wd->step;
419 if (wd->spin) ecore_timer_del(wd->spin);
420 wd->spin = ecore_timer_add(wd->interval, _spin_value, obj);
425 _val_inc_stop(Evas_Object *obj)
427 Widget_Data *wd = elm_widget_data_get(obj);
429 wd->interval = wd->first_interval;
431 if (wd->spin) ecore_timer_del(wd->spin);
436 _val_dec_start(Evas_Object *obj)
438 Widget_Data *wd = elm_widget_data_get(obj);
440 wd->interval = wd->first_interval;
441 wd->spin_speed = -wd->step;
442 if (wd->spin) ecore_timer_del(wd->spin);
443 wd->spin = ecore_timer_add(wd->interval, _spin_value, obj);
448 _val_dec_stop(Evas_Object *obj)
450 Widget_Data *wd = elm_widget_data_get(obj);
452 wd->interval = wd->first_interval;
454 if (wd->spin) ecore_timer_del(wd->spin);
459 _button_inc_start(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
461 Widget_Data *wd = elm_widget_data_get(data);
463 if (wd->entry_visible)
468 _val_inc_start(data);
472 _button_inc_stop(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
474 Widget_Data *wd = elm_widget_data_get(data);
480 _button_dec_start(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
482 Widget_Data *wd = elm_widget_data_get(data);
484 if (wd->entry_visible)
489 _val_dec_start(data);
493 _button_dec_stop(void *data, Evas_Object *obj __UNUSED__, const char *emission __UNUSED__, const char *source __UNUSED__)
495 Widget_Data *wd = elm_widget_data_get(data);
501 _entry_activated(void *data, Evas_Object *obj __UNUSED__, void *event_info __UNUSED__)
503 Widget_Data *wd = elm_widget_data_get(data);
505 _apply_entry_value(data);
506 evas_object_smart_callback_call(data, SIG_CHANGED, NULL);
507 if (wd->delay) ecore_timer_del(wd->delay);
508 wd->delay = ecore_timer_add(0.2, _delay_change, data);
512 _event_hook(Evas_Object *obj, Evas_Object *src __UNUSED__, Evas_Callback_Type type, void *event_info)
514 Widget_Data *wd = elm_widget_data_get(obj);
515 if (!wd) return EINA_FALSE;
516 if (elm_widget_disabled_get(obj)) return EINA_FALSE;
517 if (type == EVAS_CALLBACK_KEY_DOWN)
519 Evas_Event_Key_Down *ev = event_info;
520 if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) return EINA_FALSE;
521 else if (!strcmp(ev->keyname, "Left") || !strcmp(ev->keyname, "KP_Left")
522 || !strcmp(ev->keyname, "Down") || !strcmp(ev->keyname, "KP_Down"))
525 edje_object_signal_emit(wd->spinner, "elm,left,anim,activate", "elm");
526 ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
529 else if (!strcmp(ev->keyname, "Right") || !strcmp(ev->keyname, "KP_Right")
530 || !strcmp(ev->keyname, "Up") || !strcmp(ev->keyname, "KP_Up"))
533 edje_object_signal_emit(wd->spinner, "elm,right,anim,activate", "elm");
534 ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
538 else if (type == EVAS_CALLBACK_KEY_UP)
540 Evas_Event_Key_Down *ev = event_info;
541 if (ev->event_flags & EVAS_EVENT_FLAG_ON_HOLD) return EINA_FALSE;
542 if (!strcmp(ev->keyname, "Right") || !strcmp(ev->keyname, "KP_Right")
543 || !strcmp(ev->keyname, "Up") || !strcmp(ev->keyname, "KP_Up"))
545 else if (!strcmp(ev->keyname, "Left") || !strcmp(ev->keyname, "KP_Left")
546 || !strcmp(ev->keyname, "Down") || !strcmp(ev->keyname, "KP_Down"))
548 else return EINA_FALSE;
549 ev->event_flags |= EVAS_EVENT_FLAG_ON_HOLD;
556 elm_spinner_add(Evas_Object *parent)
562 ELM_WIDGET_STANDARD_SETUP(wd, Widget_Data, parent, e, obj, NULL);
564 ELM_SET_WIDTYPE(widtype, "spinner");
565 elm_widget_type_set(obj, "spinner");
566 elm_widget_sub_object_add(parent, obj);
567 elm_widget_on_focus_hook_set(obj, _on_focus_hook, NULL);
568 elm_widget_data_set(obj, wd);
569 elm_widget_del_hook_set(obj, _del_hook);
570 elm_widget_on_focus_hook_set(obj, _on_focus_hook, NULL);
571 elm_widget_theme_hook_set(obj, _theme_hook);
572 elm_widget_disable_hook_set(obj, _disable_hook);
573 elm_widget_signal_emit_hook_set(obj, _signal_emit_hook);
574 elm_widget_signal_callback_add_hook_set(obj, _signal_callback_add_hook);
575 elm_widget_signal_callback_del_hook_set(obj, _signal_callback_del_hook);
576 elm_widget_can_focus_set(obj, EINA_TRUE);
577 elm_widget_event_hook_set(obj, _event_hook);
584 wd->first_interval = 0.85;
585 wd->entry_visible = 0;
586 wd->editable = EINA_TRUE;
588 wd->spinner = edje_object_add(e);
589 _elm_theme_object_set(obj, wd->spinner, "spinner", "base", "default");
590 elm_widget_resize_object_set(obj, wd->spinner);
591 edje_object_signal_callback_add(wd->spinner, "drag", "*", _drag, obj);
592 edje_object_signal_callback_add(wd->spinner, "drag,start", "*",
594 edje_object_signal_callback_add(wd->spinner, "drag,stop", "*",
596 edje_object_signal_callback_add(wd->spinner, "drag,step", "*",
598 edje_object_signal_callback_add(wd->spinner, "drag,page", "*",
601 edje_object_signal_callback_add(wd->spinner, "elm,action,increment,start",
602 "*", _button_inc_start, obj);
603 edje_object_signal_callback_add(wd->spinner, "elm,action,increment,stop",
604 "*", _button_inc_stop, obj);
605 edje_object_signal_callback_add(wd->spinner, "elm,action,decrement,start",
606 "*", _button_dec_start, obj);
607 edje_object_signal_callback_add(wd->spinner, "elm,action,decrement,stop",
608 "*", _button_dec_stop, obj);
609 edje_object_part_drag_value_set(wd->spinner, "elm.dragable.slider",
612 wd->ent = elm_entry_add(obj);
613 elm_entry_single_line_set(wd->ent, 1);
614 evas_object_smart_callback_add(wd->ent, "activated", _entry_activated, obj);
615 edje_object_part_swallow(wd->spinner, "elm.swallow.entry", wd->ent);
616 edje_object_signal_callback_add(wd->spinner, "elm,action,entry,toggle",
617 "*", _toggle_entry, obj);
619 evas_object_smart_callbacks_descriptions_set(obj, _signals);
621 _mirrored_set(obj, elm_widget_mirrored_get(obj));
628 elm_spinner_label_format_set(Evas_Object *obj, const char *fmt)
630 ELM_CHECK_WIDTYPE(obj, widtype);
631 Widget_Data *wd = elm_widget_data_get(obj);
633 eina_stringshare_replace(&wd->label, fmt);
639 elm_spinner_label_format_get(const Evas_Object *obj)
641 ELM_CHECK_WIDTYPE(obj, widtype) NULL;
642 Widget_Data *wd = elm_widget_data_get(obj);
643 if (!wd) return NULL;
648 elm_spinner_min_max_set(Evas_Object *obj, double min, double max)
650 ELM_CHECK_WIDTYPE(obj, widtype);
651 Widget_Data *wd = elm_widget_data_get(obj);
653 if ((wd->val_min == min) && (wd->val_max == max)) return;
656 if (wd->val < wd->val_min) wd->val = wd->val_min;
657 if (wd->val > wd->val_max) wd->val = wd->val_max;
663 elm_spinner_min_max_get(const Evas_Object *obj, double *min, double *max)
667 ELM_CHECK_WIDTYPE(obj, widtype);
668 Widget_Data *wd = elm_widget_data_get(obj);
670 if (min) *min = wd->val_min;
671 if (max) *max = wd->val_max;
675 elm_spinner_step_set(Evas_Object *obj, double step)
677 ELM_CHECK_WIDTYPE(obj, widtype);
678 Widget_Data *wd = elm_widget_data_get(obj);
684 elm_spinner_step_get(const Evas_Object *obj)
686 ELM_CHECK_WIDTYPE(obj, widtype) 0.0;
687 Widget_Data *wd = elm_widget_data_get(obj);
693 elm_spinner_value_set(Evas_Object *obj, double val)
695 ELM_CHECK_WIDTYPE(obj, widtype);
696 Widget_Data *wd = elm_widget_data_get(obj);
698 if (wd->val == val) return;
700 if (wd->val < wd->val_min) wd->val = wd->val_min;
701 if (wd->val > wd->val_max) wd->val = wd->val_max;
707 elm_spinner_value_get(const Evas_Object *obj)
709 ELM_CHECK_WIDTYPE(obj, widtype) 0.0;
710 Widget_Data *wd = elm_widget_data_get(obj);
716 elm_spinner_wrap_set(Evas_Object *obj, Eina_Bool wrap)
718 ELM_CHECK_WIDTYPE(obj, widtype);
719 Widget_Data *wd = elm_widget_data_get(obj);
725 elm_spinner_wrap_get(const Evas_Object *obj)
727 ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
728 Widget_Data *wd = elm_widget_data_get(obj);
729 if (!wd) return EINA_FALSE;
734 elm_spinner_special_value_add(Evas_Object *obj, double value, const char *label)
736 Elm_Spinner_Special_Value *sv;
737 ELM_CHECK_WIDTYPE(obj, widtype);
738 Widget_Data *wd = elm_widget_data_get(obj);
741 sv = calloc(1, sizeof(*sv));
744 sv->label = eina_stringshare_add(label);
746 wd->special_values = eina_list_append(wd->special_values, sv);
751 elm_spinner_editable_set(Evas_Object *obj, Eina_Bool editable)
753 ELM_CHECK_WIDTYPE(obj, widtype);
754 Widget_Data *wd = elm_widget_data_get(obj);
756 wd->editable = editable;
760 elm_spinner_editable_get(const Evas_Object *obj)
762 ELM_CHECK_WIDTYPE(obj, widtype) EINA_FALSE;
763 Widget_Data *wd = elm_widget_data_get(obj);
764 if (!wd) return EINA_FALSE;
769 elm_spinner_interval_set(Evas_Object *obj, double interval)
771 ELM_CHECK_WIDTYPE(obj, widtype);
772 Widget_Data *wd = elm_widget_data_get(obj);
774 wd->first_interval = interval;
778 elm_spinner_interval_get(const Evas_Object *obj)
780 ELM_CHECK_WIDTYPE(obj, widtype) 0.0;
781 Widget_Data *wd = elm_widget_data_get(obj);
783 return wd->first_interval;
787 elm_spinner_base_set(Evas_Object *obj, double base)
789 ELM_CHECK_WIDTYPE(obj, widtype);
790 Widget_Data *wd = elm_widget_data_get(obj);
796 elm_spinner_base_get(const Evas_Object *obj)
798 ELM_CHECK_WIDTYPE(obj, widtype) 0.0;
799 Widget_Data *wd = elm_widget_data_get(obj);
805 elm_spinner_round_set(Evas_Object *obj, int rnd)
807 ELM_CHECK_WIDTYPE(obj, widtype);
808 Widget_Data *wd = elm_widget_data_get(obj);
814 elm_spinner_round_get(const Evas_Object *obj)
816 ELM_CHECK_WIDTYPE(obj, widtype) 0;
817 Widget_Data *wd = elm_widget_data_get(obj);