tb_inline_select.cpp 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  1. // ================================================================================
  2. // == This file is a part of Turbo Badger. (C) 2011-2014, Emil Segerås ==
  3. // == See tb_core.h for more information. ==
  4. // ================================================================================
  5. #include "tb_inline_select.h"
  6. #include <assert.h>
  7. #include <math.h>
  8. #include <stdlib.h>
  9. namespace tb {
  10. // FIX: axis should affect the buttons arrow skin!
  11. // FIX: unfocus should set the correct text!
  12. // == TBInlineSelect ========================================================================================
  13. TBInlineSelect::TBInlineSelect()
  14. : m_value(0)
  15. , m_min(0)
  16. , m_max(100)
  17. , m_modified(false)
  18. {
  19. SetSkinBg(TBIDC("TBInlineSelect"));
  20. AddChild(&m_layout);
  21. m_layout.AddChild(&m_buttons[0]);
  22. m_layout.AddChild(&m_editfield);
  23. m_layout.AddChild(&m_buttons[1]);
  24. m_layout.SetRect(GetPaddingRect());
  25. m_layout.SetGravity(WIDGET_GRAVITY_ALL);
  26. m_layout.SetSpacing(0);
  27. m_buttons[0].SetSkinBg(TBIDC("TBButton.flat"));
  28. m_buttons[1].SetSkinBg(TBIDC("TBButton.flat"));
  29. m_buttons[0].GetContentRoot()->AddChild(new TBSkinImage(TBIDC("arrowdark.left")));
  30. m_buttons[1].GetContentRoot()->AddChild(new TBSkinImage(TBIDC("arrowdark.right")));
  31. m_buttons[0].SetIsFocusable(false);
  32. m_buttons[1].SetIsFocusable(false);
  33. m_buttons[0].SetID(TBIDC("dec"));
  34. m_buttons[1].SetID(TBIDC("inc"));
  35. m_buttons[0].SetAutoRepeat(true);
  36. m_buttons[1].SetAutoRepeat(true);
  37. m_editfield.SetID(TBIDC("edit"));
  38. m_editfield.SetTextAlign(TB_TEXT_ALIGN_CENTER);
  39. m_editfield.SetEditType(EDIT_TYPE_NUMBER);
  40. m_editfield.SetText("0");
  41. m_editfield.AddListener(this);
  42. }
  43. TBInlineSelect::~TBInlineSelect()
  44. {
  45. m_editfield.RemoveListener(this);
  46. m_layout.RemoveChild(&m_buttons[1]);
  47. m_layout.RemoveChild(&m_editfield);
  48. m_layout.RemoveChild(&m_buttons[0]);
  49. RemoveChild(&m_layout);
  50. }
  51. void TBInlineSelect::SetEditFieldLayoutParams(LayoutParams& lp)
  52. {
  53. m_editfield.SetLayoutParams(lp);
  54. }
  55. void TBInlineSelect::SetLimits(double min, double max)
  56. {
  57. assert(min <= max);
  58. m_min = min;
  59. m_max = max;
  60. SetValueDouble(m_value);
  61. }
  62. void TBInlineSelect::SetValueInternal(double value, bool update_text)
  63. {
  64. value = CLAMP(value, m_min, m_max);
  65. if (value == m_value)
  66. return;
  67. m_value = value;
  68. if (update_text)
  69. {
  70. TBStr strval;
  71. double prec = m_value - floor(m_value);
  72. if (prec < .001)
  73. {
  74. strval.SetFormatted("%.0f", m_value);
  75. }
  76. else
  77. strval.SetFormatted("%.2f", m_value);
  78. m_editfield.SetText(strval);
  79. }
  80. TBWidgetEvent ev(EVENT_TYPE_CHANGED);
  81. InvokeEvent(ev);
  82. // Warning: Do nothing here since the event might have deleted us.
  83. // If needed, check if we are alive using a safe pointer first.
  84. }
  85. void TBInlineSelect::OnSkinChanged()
  86. {
  87. m_layout.SetRect(GetPaddingRect());
  88. }
  89. bool TBInlineSelect::OnEvent(const TBWidgetEvent &ev)
  90. {
  91. if (ev.type == EVENT_TYPE_KEY_DOWN)
  92. {
  93. if (ev.special_key == TB_KEY_UP || ev.special_key == TB_KEY_DOWN)
  94. {
  95. double dv = ev.special_key == TB_KEY_UP ? 1 : -1;
  96. SetValueDouble(GetValueDouble() + dv);
  97. return true;
  98. }
  99. }
  100. else if (ev.type == EVENT_TYPE_CLICK && ev.target->GetID() == TBIDC("dec"))
  101. {
  102. SetValueDouble(GetValueDouble() - 1);
  103. if (!ev.target->IsCaptured()) {
  104. InvokeModifiedEvent();
  105. }
  106. return true;
  107. }
  108. else if (ev.type == EVENT_TYPE_CLICK && ev.target->GetID() == TBIDC("inc"))
  109. {
  110. SetValueDouble(GetValueDouble() + 1);
  111. if (!ev.target->IsCaptured()) {
  112. InvokeModifiedEvent();
  113. }
  114. return true;
  115. }
  116. else if (ev.type == EVENT_TYPE_CHANGED && ev.target == &m_editfield)
  117. {
  118. TBStr text;
  119. m_editfield.GetText(text);
  120. SetValueInternal((double) atof(text), false);
  121. }
  122. else if (ev.type == EVENT_TYPE_CHANGED && ev.target == this)
  123. {
  124. return TBWidget::OnEvent(ev);
  125. }
  126. return false;
  127. }
  128. void TBInlineSelect::InvokeModifiedEvent()
  129. {
  130. TBWidgetEvent ev(EVENT_TYPE_CUSTOM);
  131. // TBIDC does not register the TBID with the UI system, so do it this way
  132. TBID refid("edit_complete");
  133. ev.ref_id = refid;
  134. // forward to delegate
  135. TBWidget::OnEvent(ev);
  136. m_modified = false;
  137. m_editfield.GetText(m_initial_edit_value);
  138. }
  139. bool TBInlineSelect::OnWidgetInvokeEvent(TBWidget *widget, const TBWidgetEvent &ev)
  140. {
  141. return false;
  142. }
  143. void TBInlineSelect::OnWidgetFocusChanged(TBWidget *widget, bool focused)
  144. {
  145. if (widget == &m_editfield)
  146. {
  147. if (focused)
  148. m_editfield.GetText(m_initial_edit_value);
  149. else
  150. {
  151. TBStr editvalue;
  152. m_editfield.GetText(editvalue);
  153. if (m_modified || !editvalue.Equals(m_initial_edit_value.CStr()))
  154. {
  155. InvokeModifiedEvent();
  156. }
  157. }
  158. }
  159. }
  160. }; // namespace tb