ScrollBar.cpp 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326
  1. //
  2. // Copyright (c) 2008-2013 the Urho3D project.
  3. //
  4. // Permission is hereby granted, free of charge, to any person obtaining a copy
  5. // of this software and associated documentation files (the "Software"), to deal
  6. // in the Software without restriction, including without limitation the rights
  7. // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  8. // copies of the Software, and to permit persons to whom the Software is
  9. // furnished to do so, subject to the following conditions:
  10. //
  11. // The above copyright notice and this permission notice shall be included in
  12. // all copies or substantial portions of the Software.
  13. //
  14. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  19. // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  20. // THE SOFTWARE.
  21. //
  22. #include "Precompiled.h"
  23. #include "Button.h"
  24. #include "Context.h"
  25. #include "InputEvents.h"
  26. #include "ScrollBar.h"
  27. #include "Slider.h"
  28. #include "UIEvents.h"
  29. namespace Urho3D
  30. {
  31. static const float DEFAULT_SCROLL_STEP = 0.1f;
  32. static const float DEFAULT_REPEAT_DELAY = 0.4f;
  33. static const float DEFAULT_REPEAT_RATE = 20.0f;
  34. extern const char* orientations[];
  35. extern const char* UI_CATEGORY;
  36. ScrollBar::ScrollBar(Context* context) :
  37. UIElement(context),
  38. scrollStep_(DEFAULT_SCROLL_STEP),
  39. stepFactor_(1.0f),
  40. leftRect_(IntRect::ZERO),
  41. rightRect_(IntRect::ZERO),
  42. upRect_(IntRect::ZERO),
  43. downRect_(IntRect::ZERO)
  44. {
  45. enabled_ = true;
  46. backButton_ = CreateChild<Button>("SB_Back");
  47. backButton_->SetInternal(true);
  48. backButton_->SetRepeat(DEFAULT_REPEAT_DELAY, DEFAULT_REPEAT_RATE);
  49. slider_ = CreateChild<Slider>("SB_Slider");
  50. slider_->SetInternal(true);
  51. slider_->SetRepeatRate(DEFAULT_REPEAT_RATE);
  52. forwardButton_ = CreateChild<Button>("SB_Forward");
  53. forwardButton_->SetInternal(true);
  54. forwardButton_->SetRepeat(DEFAULT_REPEAT_DELAY, DEFAULT_REPEAT_RATE);
  55. SubscribeToEvent(backButton_, E_PRESSED, HANDLER(ScrollBar, HandleBackButtonPressed));
  56. SubscribeToEvent(forwardButton_, E_PRESSED, HANDLER(ScrollBar, HandleForwardButtonPressed));
  57. SubscribeToEvent(slider_, E_SLIDERCHANGED, HANDLER(ScrollBar, HandleSliderChanged));
  58. SubscribeToEvent(slider_, E_SLIDERPAGED, HANDLER(ScrollBar, HandleSliderPaged));
  59. // Set default orientation/layout
  60. SetOrientation(O_HORIZONTAL);
  61. }
  62. ScrollBar::~ScrollBar()
  63. {
  64. }
  65. void ScrollBar::RegisterObject(Context* context)
  66. {
  67. context->RegisterFactory<ScrollBar>(UI_CATEGORY);
  68. COPY_BASE_ATTRIBUTES(ScrollBar, UIElement);
  69. UPDATE_ATTRIBUTE_DEFAULT_VALUE(ScrollBar, "Is Enabled", true);
  70. ENUM_ACCESSOR_ATTRIBUTE(ScrollBar, "Orientation", GetOrientation, SetOrientation, Orientation, orientations, O_HORIZONTAL, AM_FILE);
  71. ACCESSOR_ATTRIBUTE(ScrollBar, VAR_FLOAT, "Range", GetRange, SetRange, float, 1.0f, AM_FILE);
  72. ACCESSOR_ATTRIBUTE(ScrollBar, VAR_FLOAT, "Value", GetValue, SetValue, float, 0.0f, AM_FILE);
  73. ACCESSOR_ATTRIBUTE(ScrollBar, VAR_FLOAT, "Scroll Step", GetScrollStep, SetScrollStep, float, DEFAULT_SCROLL_STEP, AM_FILE);
  74. ACCESSOR_ATTRIBUTE(ScrollBar, VAR_FLOAT, "Step Factor", GetStepFactor, SetStepFactor, float, 1.0f, AM_FILE);
  75. ATTRIBUTE(ScrollBar, VAR_INTRECT, "Left Image Rect", leftRect_, IntRect::ZERO, AM_FILE);
  76. ATTRIBUTE(ScrollBar, VAR_INTRECT, "Right Image Rect", rightRect_, IntRect::ZERO, AM_FILE);
  77. ATTRIBUTE(ScrollBar, VAR_INTRECT, "Up Image Rect", upRect_, IntRect::ZERO, AM_FILE);
  78. ATTRIBUTE(ScrollBar, VAR_INTRECT, "Down Image Rect", downRect_, IntRect::ZERO, AM_FILE);
  79. }
  80. void ScrollBar::ApplyAttributes()
  81. {
  82. UIElement::ApplyAttributes();
  83. // Reapply orientation to the button images
  84. if (slider_->GetOrientation() == O_HORIZONTAL)
  85. {
  86. backButton_->SetImageRect(leftRect_);
  87. forwardButton_->SetImageRect(rightRect_);
  88. }
  89. else
  90. {
  91. backButton_->SetImageRect(upRect_);
  92. forwardButton_->SetImageRect(downRect_);
  93. }
  94. }
  95. void ScrollBar::OnResize()
  96. {
  97. // Disable layout operations while setting the button sizes is incomplete
  98. DisableLayoutUpdate();
  99. if (slider_->GetOrientation() == O_HORIZONTAL)
  100. {
  101. int height = GetHeight();
  102. backButton_->SetFixedSize(height, height);
  103. forwardButton_->SetFixedSize(height, height);
  104. }
  105. else
  106. {
  107. int width = GetWidth();
  108. backButton_->SetFixedSize(width, width);
  109. forwardButton_->SetFixedSize(width, width);
  110. }
  111. EnableLayoutUpdate();
  112. }
  113. void ScrollBar::OnSetEditable()
  114. {
  115. slider_->SetEditable(editable_);
  116. }
  117. void ScrollBar::SetOrientation(Orientation orientation)
  118. {
  119. slider_->SetOrientation(orientation);
  120. if (orientation == O_HORIZONTAL)
  121. {
  122. backButton_->SetImageRect(leftRect_);
  123. forwardButton_->SetImageRect(rightRect_);
  124. }
  125. else
  126. {
  127. backButton_->SetImageRect(upRect_);
  128. forwardButton_->SetImageRect(downRect_);
  129. }
  130. OnResize();
  131. if (orientation == O_HORIZONTAL)
  132. SetLayout(LM_HORIZONTAL);
  133. else
  134. SetLayout(LM_VERTICAL);
  135. }
  136. void ScrollBar::SetRange(float range)
  137. {
  138. slider_->SetRange(range);
  139. }
  140. void ScrollBar::SetValue(float value)
  141. {
  142. slider_->SetValue(value);
  143. }
  144. void ScrollBar::ChangeValue(float delta)
  145. {
  146. slider_->ChangeValue(delta);
  147. }
  148. void ScrollBar::SetScrollStep(float step)
  149. {
  150. scrollStep_ = Max(step, 0.0f);
  151. }
  152. void ScrollBar::SetStepFactor(float factor)
  153. {
  154. stepFactor_ = Max(factor, M_EPSILON);
  155. }
  156. void ScrollBar::StepBack()
  157. {
  158. slider_->SetValue(slider_->GetValue() - GetEffectiveScrollStep());
  159. }
  160. void ScrollBar::StepForward()
  161. {
  162. slider_->SetValue(slider_->GetValue() + GetEffectiveScrollStep());
  163. }
  164. Orientation ScrollBar::GetOrientation() const
  165. {
  166. return slider_->GetOrientation();
  167. }
  168. float ScrollBar::GetRange() const
  169. {
  170. return slider_->GetRange();
  171. }
  172. float ScrollBar::GetValue() const
  173. {
  174. return slider_->GetValue();
  175. }
  176. float ScrollBar::GetEffectiveScrollStep() const
  177. {
  178. return scrollStep_ * stepFactor_;
  179. }
  180. bool ScrollBar::FilterImplicitAttributes(XMLElement& dest) const
  181. {
  182. if (!UIElement::FilterImplicitAttributes(dest))
  183. return false;
  184. if (!RemoveChildXML(dest, "Layout Mode"))
  185. return false;
  186. XMLElement childElem = dest.GetChild("element");
  187. if (!FilterButtonImplicitAttributes(childElem, "SB_Back"))
  188. return false;
  189. childElem = childElem.GetNext("element");
  190. if (!childElem)
  191. return false;
  192. if (!RemoveChildXML(childElem, "Name", "SB_Slider"))
  193. return false;
  194. if (!RemoveChildXML(childElem, "Repeat Rate", String(DEFAULT_REPEAT_RATE)))
  195. return false;
  196. if (!RemoveChildXML(childElem, "Orientation"))
  197. return false;
  198. if (!RemoveChildXML(childElem, "Range"))
  199. return false;
  200. if (!RemoveChildXML(childElem, "Value"))
  201. return false;
  202. childElem = childElem.GetNext("element");
  203. if (!FilterButtonImplicitAttributes(childElem, "SB_Forward"))
  204. return false;
  205. return true;
  206. }
  207. bool ScrollBar::FilterButtonImplicitAttributes(XMLElement& dest, const String& name) const
  208. {
  209. if (!dest)
  210. return false;
  211. if (!RemoveChildXML(dest, "Name", name))
  212. return false;
  213. if (!RemoveChildXML(dest, "Repeat Delay", String(DEFAULT_REPEAT_DELAY)))
  214. return false;
  215. if (!RemoveChildXML(dest, "Repeat Rate", String(DEFAULT_REPEAT_RATE)))
  216. return false;
  217. if (!RemoveChildXML(dest, "Image Rect"))
  218. return false;
  219. if (!RemoveChildXML(dest, "Min Size"))
  220. return false;
  221. if (!RemoveChildXML(dest, "Max Size"))
  222. return false;
  223. return true;
  224. }
  225. void ScrollBar::HandleBackButtonPressed(StringHash eventType, VariantMap& eventData)
  226. {
  227. if (editable_)
  228. StepBack();
  229. }
  230. void ScrollBar::HandleForwardButtonPressed(StringHash eventType, VariantMap& eventData)
  231. {
  232. if (editable_)
  233. StepForward();
  234. }
  235. void ScrollBar::HandleSliderChanged(StringHash eventType, VariantMap& eventData)
  236. {
  237. // Send the event forward
  238. VariantMap newEventData;
  239. newEventData[ScrollBarChanged::P_ELEMENT] = (void*)this;
  240. newEventData[ScrollBarChanged::P_VALUE] = slider_->GetValue();
  241. SendEvent(E_SCROLLBARCHANGED, newEventData);
  242. }
  243. void ScrollBar::HandleSliderPaged(StringHash eventType, VariantMap& eventData)
  244. {
  245. using namespace SliderPaged;
  246. // Synthesize hover event to the forward/back buttons
  247. if (eventData[P_OFFSET].GetInt() < 0)
  248. backButton_->OnHover(IntVector2::ZERO, backButton_->ElementToScreen(IntVector2::ZERO), 0, 0, 0);
  249. else
  250. forwardButton_->OnHover(IntVector2::ZERO, forwardButton_->ElementToScreen(IntVector2::ZERO), 0, 0, 0);
  251. // Synthesize click / release events to the buttons
  252. if (eventData[P_PRESSED].GetBool())
  253. {
  254. if (eventData[P_OFFSET].GetInt() < 0)
  255. {
  256. backButton_->OnClickBegin(IntVector2::ZERO, backButton_->ElementToScreen(IntVector2::ZERO), MOUSEB_LEFT, MOUSEB_LEFT,
  257. 0, 0);
  258. }
  259. else
  260. {
  261. forwardButton_->OnClickBegin(IntVector2::ZERO, forwardButton_->ElementToScreen(IntVector2::ZERO), MOUSEB_LEFT,
  262. MOUSEB_LEFT, 0, 0);
  263. }
  264. }
  265. else
  266. {
  267. if (eventData[P_OFFSET].GetInt() < 0)
  268. {
  269. backButton_->OnClickEnd(IntVector2::ZERO, backButton_->ElementToScreen(IntVector2::ZERO), MOUSEB_LEFT, 0, 0, 0,
  270. backButton_);
  271. }
  272. else
  273. {
  274. forwardButton_->OnClickEnd(IntVector2::ZERO, forwardButton_->ElementToScreen(IntVector2::ZERO), MOUSEB_LEFT, 0, 0, 0,
  275. forwardButton_);
  276. }
  277. }
  278. }
  279. }