d122835fcb49e6f67b37c828011ee4ad7e781c8c
[platform/core/uifw/dali-toolkit.git] / dali-toolkit / internal / controls / scrollable / scrollable-impl.cpp
1 //
2 // Copyright (c) 2014 Samsung Electronics Co., Ltd.
3 //
4 // Licensed under the Flora License, Version 1.0 (the License);
5 // you may not use this file except in compliance with the License.
6 // You may obtain a copy of the License at
7 //
8 //     http://floralicense.org/license/
9 //
10 // Unless required by applicable law or agreed to in writing, software
11 // distributed under the License is distributed on an AS IS BASIS,
12 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 // See the License for the specific language governing permissions and
14 // limitations under the License.
15 //
16
17 // INTERNAL INCLUDES
18 #include <dali-toolkit/internal/controls/scrollable/scrollable-impl.h>
19 #include <dali-toolkit/internal/controls/scroll-component/scroll-bar-internal-impl.h>
20
21 using namespace Dali;
22
23 namespace
24 {
25
26 } // unnamed namespace
27
28 namespace Dali
29 {
30
31 namespace Toolkit
32 {
33
34 namespace Internal
35 {
36
37 namespace
38 {
39
40 BaseHandle Create()
41 {
42   // empty handle as we cannot create Scrollable (but type registered for scroll signal)
43   return BaseHandle();
44 }
45
46 TypeRegistration mType( typeid(Toolkit::Scrollable), typeid(Toolkit::Control), Create );
47
48 SignalConnectorType s1(mType, Toolkit::Scrollable::SIGNAL_SCROLL_STARTED,   &Scrollable::DoConnectSignal);
49 SignalConnectorType s2(mType, Toolkit::Scrollable::SIGNAL_SCROLL_COMPLETED, &Scrollable::DoConnectSignal);
50 SignalConnectorType s3(mType, Toolkit::Scrollable::SIGNAL_SCROLL_UPDATED,   &Scrollable::DoConnectSignal);
51 SignalConnectorType s4(mType, Toolkit::Scrollable::SIGNAL_SCROLL_CLAMPED,   &Scrollable::DoConnectSignal);
52
53 }
54
55 const std::string Scrollable::SCROLLABLE_CAN_SCROLL_VERTICAL( "scrollable-can-scroll-vertical" );
56 const std::string Scrollable::SCROLLABLE_CAN_SCROLL_HORIZONTAL( "scrollable-can-scroll-horizontal" );
57
58 ///////////////////////////////////////////////////////////////////////////////////////////////////
59 // Scrollable
60 ///////////////////////////////////////////////////////////////////////////////////////////////////
61
62 Scrollable::Scrollable()
63 : ControlImpl(true/*requires touch*/),
64   mPropertyRelativePosition(Property::INVALID_INDEX),
65   mPropertyPositionMin(Property::INVALID_INDEX),
66   mPropertyPositionMax(Property::INVALID_INDEX),
67   mPropertyScrollDirection(Property::INVALID_INDEX),
68   mOvershootEnabled(false)
69 {
70 }
71
72 Scrollable::~Scrollable()
73 {
74   // Clear scroll components, forces their destruction before Scrollable is destroyed.
75   mComponents.clear();
76 }
77
78 void Scrollable::RegisterCommonProperties()
79 {
80   Actor self = Self();
81
82   // Register properties.
83   mPropertyRelativePosition = self.RegisterProperty(Toolkit::Scrollable::SCROLL_RELATIVE_POSITION_PROPERTY_NAME, Vector3::ZERO);
84   mPropertyPositionMin = self.RegisterProperty(Toolkit::Scrollable::SCROLL_POSITION_MIN_PROPERTY_NAME, Vector3::ZERO);
85   mPropertyPositionMax = self.RegisterProperty(Toolkit::Scrollable::SCROLL_POSITION_MAX_PROPERTY_NAME, Vector3::ZERO);
86   mPropertyScrollDirection = self.RegisterProperty(Toolkit::Scrollable::SCROLL_DIRECTION_PROPERTY_NAME, Vector3::ZERO);
87   mPropertyCanScrollVertical = self.RegisterProperty(SCROLLABLE_CAN_SCROLL_VERTICAL, true);
88   mPropertyCanScrollHorizontal = self.RegisterProperty(SCROLLABLE_CAN_SCROLL_HORIZONTAL, true);
89 }
90
91 bool Scrollable::IsScrollComponentEnabled(Toolkit::Scrollable::ScrollComponentType type) const
92 {
93   if(type == Toolkit::Scrollable::OvershootIndicator)
94   {
95     return mOvershootEnabled;
96   }
97   return (mComponents.find(type) != mComponents.end());
98 }
99
100 void Scrollable::EnableScrollComponent(Toolkit::Scrollable::ScrollComponentType type)
101 {
102   if(type == Toolkit::Scrollable::OvershootIndicator)
103   {
104     SetOvershootEnabled(true);
105     mOvershootEnabled = true;
106     return;
107   }
108   if( mComponents.find(type) == mComponents.end() )
109   {
110     // Create ScrollComponent
111     Toolkit::Scrollable scrollable = Toolkit::Scrollable::DownCast(Self());
112     Toolkit::ScrollComponent scrollComponent = NewScrollComponent(scrollable, type);
113     Toolkit::ScrollComponentImpl& component = static_cast<Toolkit::ScrollComponentImpl&>(scrollComponent.GetImplementation());
114     ScrollComponentPtr componentPtr(&component);
115
116     mComponents[type] = componentPtr;
117   }
118 }
119
120 void Scrollable::DisableScrollComponent(Toolkit::Scrollable::ScrollComponentType type)
121 {
122   if(type == Toolkit::Scrollable::OvershootIndicator)
123   {
124     SetOvershootEnabled(false);
125     mOvershootEnabled = false;
126     return;
127   }
128   ComponentIter pair = mComponents.find( type );
129
130   if( mComponents.end() != pair )
131   {
132     ScrollComponentPtr component = pair->second;
133
134     // Destroy ScrollComponent.
135     mComponents.erase( type );
136   }
137 }
138
139 Toolkit::Scrollable::ScrollStartedSignalV2& Scrollable::ScrollStartedSignal()
140 {
141   return mScrollStartedSignalV2;
142 }
143
144 Toolkit::Scrollable::ScrollUpdatedSignalV2& Scrollable::ScrollUpdatedSignal()
145 {
146   return mScrollUpdatedSignalV2;
147 }
148
149 Toolkit::Scrollable::ScrollCompletedSignalV2& Scrollable::ScrollCompletedSignal()
150 {
151   return mScrollCompletedSignalV2;
152 }
153
154 Toolkit::Scrollable::ScrollClampedSignalV2& Scrollable::ScrollClampedSignal()
155 {
156   return mScrollClampedSignalV2;
157 }
158
159 bool Scrollable::DoConnectSignal( BaseObject* object, ConnectionTrackerInterface* tracker, const std::string& signalName, FunctorDelegate* functor )
160 {
161   Dali::BaseHandle handle( object );
162
163   bool connected( true );
164   Toolkit::Scrollable scrollable = Toolkit::Scrollable::DownCast( handle );
165
166   if( Toolkit::Scrollable::SIGNAL_SCROLL_STARTED == signalName )
167   {
168     scrollable.ScrollStartedSignal().Connect( tracker, functor );
169   }
170   else if( Toolkit::Scrollable::SIGNAL_SCROLL_UPDATED == signalName )
171   {
172     scrollable.ScrollUpdatedSignal().Connect( tracker, functor );
173   }
174   else if( Toolkit::Scrollable::SIGNAL_SCROLL_COMPLETED == signalName )
175   {
176     scrollable.ScrollCompletedSignal().Connect( tracker, functor );
177   }
178   else if( Toolkit::Scrollable::SIGNAL_SCROLL_CLAMPED == signalName )
179   {
180     scrollable.ScrollClampedSignal().Connect( tracker, functor );
181   }
182   else
183   {
184     // signalName does not match any signal
185     connected = false;
186   }
187
188   return connected;
189 }
190
191 Toolkit::ScrollComponent Scrollable::NewScrollComponent(Toolkit::Scrollable& scrollable, Toolkit::Scrollable::ScrollComponentType type)
192 {
193   Toolkit::ScrollComponent instance;
194
195   switch(type)
196   {
197     case Toolkit::Scrollable::VerticalScrollBar:
198     {
199       instance = static_cast<Toolkit::ScrollComponent>(Toolkit::ScrollBarInternal::New(scrollable, true));
200       break;
201     }
202     case Toolkit::Scrollable::HorizontalScrollBar:
203     {
204       instance = static_cast<Toolkit::ScrollComponent>(Toolkit::ScrollBarInternal::New(scrollable, false));
205       break;
206     }
207     default:
208     {
209       DALI_ASSERT_ALWAYS(true && "Unrecognized component type");
210     }
211   }
212
213   return instance;
214 }
215
216 } // namespace Internal
217
218 } // namespace Toolkit
219
220 } // namespace Dali