Exporting one constructor used by csharpbinder and making an API private.
[platform/core/uifw/dali-toolkit.git] / dali-toolkit / internal / layouting / vbox-layout-impl.cpp
1 /*
2  * Copyright (c) 2018 Samsung Electronics Co., Ltd.
3  *
4  * Licensed under the Apache License, Version 2.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://www.apache.org/licenses/LICENSE-2.0
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 //CLASS HEADER
18 #include <dali-toolkit/internal/layouting/vbox-layout-impl.h>
19
20 //INTERNAL HEADERS
21 #include <dali/integration-api/debug.h>
22 #include <dali/public-api/common/extents.h>
23 #include <dali/devel-api/actors/actor-devel.h>
24 #include <dali/devel-api/object/handle-devel.h>
25 #include <dali-toolkit/devel-api/layouting/layout-item.h>
26 #include <dali-toolkit/public-api/controls/control-impl.h>
27 #include <dali-toolkit/internal/controls/control/control-data-impl.h>
28
29 namespace
30 {
31 #if defined(DEBUG_ENABLED)
32 static Debug::Filter* gLogFilter = Debug::Filter::New( Debug::Concise, false, "LOG_LAYOUT" );
33 #endif
34 }
35
36 namespace Dali
37 {
38 namespace Toolkit
39 {
40 namespace Internal
41 {
42
43 VboxLayoutPtr VboxLayout::New()
44 {
45   VboxLayoutPtr layout( new VboxLayout() );
46   return layout;
47 }
48
49 VboxLayout::VboxLayout()
50 : LayoutGroup(),
51   mCellPadding( 0, 0 ),
52   mTotalLength( 0 )
53 {
54 }
55
56 VboxLayout::~VboxLayout()
57 {
58 }
59
60 void VboxLayout::DoInitialize()
61 {
62 }
63
64 void VboxLayout::DoRegisterChildProperties( const std::string& containerType )
65 {
66   auto typeInfo = Dali::TypeRegistry::Get().GetTypeInfo( containerType );
67   if( typeInfo )
68   {
69     Property::IndexContainer indices;
70     typeInfo.GetChildPropertyIndices( indices );
71
72     if( std::find( indices.Begin(), indices.End(), Toolkit::VboxLayout::ChildProperty::WEIGHT ) ==
73         indices.End() )
74     {
75       ChildPropertyRegistration( typeInfo.GetName(), "weight", Toolkit::VboxLayout::ChildProperty::WEIGHT, Property::FLOAT );
76     }
77   }
78 }
79
80 void VboxLayout::OnChildAdd( LayoutItem& child )
81 {
82   auto owner = child.GetOwner();
83   owner.SetProperty( Toolkit::VboxLayout::ChildProperty::WEIGHT, 1.0f );
84 }
85
86 void VboxLayout::SetCellPadding( LayoutSize size )
87 {
88   mCellPadding = size;
89 }
90
91 LayoutSize VboxLayout::GetCellPadding()
92 {
93   return mCellPadding;
94 }
95
96
97 void VboxLayout::OnMeasure( MeasureSpec widthMeasureSpec, MeasureSpec heightMeasureSpec )
98 {
99 #if defined(DEBUG_ENABLED)
100   auto actor = Actor::DownCast(GetOwner());
101
102   std::ostringstream oss;
103   oss << "VboxLayout::OnMeasure  ";
104   if( actor )
105   {
106     oss << "Actor Id:" << actor.GetId() << " Name:" << actor.GetName() << "  ";
107   }
108   oss << "widthMeasureSpec:" << widthMeasureSpec << " heightMeasureSpec:" << heightMeasureSpec << std::endl;
109   DALI_LOG_INFO( gLogFilter, Debug::Concise, oss.str().c_str() );
110 #endif
111
112   auto widthMode = widthMeasureSpec.GetMode();
113
114   bool matchWidth = false;
115   bool allFillParent = true;
116   LayoutLength maxWidth = 0;
117   LayoutLength alternativeMaxWidth = 0;
118
119   struct
120   {
121     MeasuredSize::State widthState;
122     MeasuredSize::State heightState;
123   } childState = { MeasuredSize::State::MEASURED_SIZE_OK, MeasuredSize::State::MEASURED_SIZE_OK };
124
125   // measure children, and determine if further resolution is required
126   for( unsigned int i=0; i<GetChildCount(); ++i )
127   {
128     auto childLayout = GetChildAt( i );
129     if( childLayout )
130     {
131       auto childOwner = childLayout->GetOwner();
132       auto desiredWidth = childOwner.GetProperty<int>( Toolkit::LayoutItem::ChildProperty::WIDTH_SPECIFICATION );
133
134       MeasureChildWithMargins( childLayout, widthMeasureSpec, 0, heightMeasureSpec, 0 );
135       auto childHeight = childLayout->GetMeasuredHeight();
136       auto childMargin = childOwner.GetProperty<Extents>( Toolkit::LayoutGroup::ChildProperty::MARGIN_SPECIFICATION );
137       auto length = childHeight + LayoutLength::IntType(childMargin.top + childMargin.bottom );
138
139       auto cellPadding = i<GetChildCount()-1 ? mCellPadding.height : 0;
140       auto totalLength = mTotalLength;
141       mTotalLength = std::max( totalLength, totalLength + length + cellPadding);
142
143       bool matchWidthLocally = false;
144       if( widthMode != MeasureSpec::Mode::EXACTLY && desiredWidth == Toolkit::ChildLayoutData::MATCH_PARENT )
145       {
146         // Will have to re-measure at least this child when we know exact height.
147         matchWidth = true;
148         matchWidthLocally = true;
149       }
150
151       auto marginWidth = LayoutLength( childMargin.start + childMargin.end );
152       auto childWidth = childLayout->GetMeasuredWidth() + marginWidth;
153
154       // was combineMeasuredStates()
155       if( childLayout->GetMeasuredWidthAndState().GetState() == MeasuredSize::State::MEASURED_SIZE_TOO_SMALL )
156       {
157         childState.widthState = MeasuredSize::State::MEASURED_SIZE_TOO_SMALL;
158       }
159       if( childLayout->GetMeasuredHeightAndState().GetState() == MeasuredSize::State::MEASURED_SIZE_TOO_SMALL )
160       {
161         childState.heightState = MeasuredSize::State::MEASURED_SIZE_TOO_SMALL;
162       }
163
164       maxWidth = std::max( maxWidth, childWidth );
165       allFillParent = ( allFillParent && desiredWidth == Toolkit::ChildLayoutData::MATCH_PARENT );
166       alternativeMaxWidth = std::max( alternativeMaxWidth, matchWidthLocally ? marginWidth : childWidth );
167     }
168   }
169   Extents padding = GetPadding();
170   mTotalLength += padding.top + padding.bottom;
171   auto heightSize = mTotalLength;
172   heightSize = std::max( heightSize, GetSuggestedMinimumHeight() );
173   MeasuredSize heightSizeAndState = ResolveSizeAndState( heightSize, heightMeasureSpec, MeasuredSize::State::MEASURED_SIZE_OK);
174   heightSize = heightSizeAndState.GetSize();
175
176   if( !allFillParent && widthMode != MeasureSpec::Mode::EXACTLY )
177   {
178     maxWidth = alternativeMaxWidth;
179   }
180   maxWidth += padding.start + padding.end;
181   maxWidth = std::max( maxWidth, GetSuggestedMinimumWidth() );
182
183   heightSizeAndState.SetState( childState.heightState );
184
185   SetMeasuredDimensions( ResolveSizeAndState( maxWidth, widthMeasureSpec, childState.widthState ),
186                          heightSizeAndState );
187
188   if( matchWidth )
189   {
190     ForceUniformWidth( GetChildCount(), heightMeasureSpec );
191   }
192 }
193
194 void VboxLayout::ForceUniformWidth( int count, MeasureSpec heightMeasureSpec )
195 {
196   // Pretend that the linear layout has an exact size. This is the measured height of
197   // ourselves. The measured height should be the max height of the children, changed
198   // to accommodate the heightMeasureSpec from the parent
199   auto uniformMeasureSpec = MeasureSpec( GetMeasuredWidth(), MeasureSpec::Mode::EXACTLY );
200   for (int i = 0; i < count; ++i)
201   {
202     LayoutItemPtr childLayout = GetChildAt(i);
203     if( childLayout != nullptr )
204     {
205       auto childOwner = childLayout->GetOwner();
206       auto desiredWidth = childOwner.GetProperty<int>( Toolkit::LayoutItem::ChildProperty::WIDTH_SPECIFICATION );
207       auto desiredHeight = childOwner.GetProperty<int>( Toolkit::LayoutItem::ChildProperty::HEIGHT_SPECIFICATION );
208
209       if( desiredWidth == Toolkit::ChildLayoutData::MATCH_PARENT )
210       {
211         // Temporarily force children to reuse their old measured height
212         int oldHeight = desiredHeight;
213         childOwner.SetProperty( Toolkit::LayoutItem::ChildProperty::HEIGHT_SPECIFICATION, childLayout->GetMeasuredHeight().mValue );
214
215         // Remeasure with new dimensions
216         MeasureChildWithMargins( childLayout, uniformMeasureSpec, 0, heightMeasureSpec, 0 );
217
218         childOwner.SetProperty( Toolkit::LayoutItem::ChildProperty::HEIGHT_SPECIFICATION, oldHeight );
219       }
220     }
221   }
222 }
223
224 void VboxLayout::OnLayout( bool changed, LayoutLength left, LayoutLength top, LayoutLength right, LayoutLength bottom )
225 {
226   Extents padding = GetPadding();
227
228   LayoutLength childTop( 0 );
229   LayoutLength childLeft( padding.start );
230
231   // Where bottom of child should go
232   auto width = right - left;
233
234   // Space available for child
235   auto childSpace = width - padding.start - padding.end;
236   auto count = GetChildCount();
237
238   for( unsigned int childIndex = 0; childIndex < count; childIndex++)
239   {
240     LayoutItemPtr childLayout = GetChildAt( childIndex );
241     if( childLayout != nullptr )
242     {
243       auto childWidth = childLayout->GetMeasuredWidth();
244       auto childHeight = childLayout->GetMeasuredHeight();
245
246       auto childOwner = childLayout->GetOwner();
247       auto childMargin = childOwner.GetProperty<Extents>( Toolkit::LayoutGroup::ChildProperty::MARGIN_SPECIFICATION );
248
249       childTop += childMargin.top;
250       childLeft = ( childSpace - childWidth ) / 2 + childMargin.start - childMargin.end;
251
252       childLayout->Layout( childLeft, childTop, childLeft + childWidth, childTop + childHeight );
253       childTop += childHeight + childMargin.bottom + mCellPadding.height;
254     }
255   }
256 }
257
258 } // namespace Internal
259 } // namespace Toolkit
260 } // namespace Dali