Fix to use width and height correctly in flex-layout-impl.cpp
[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::NoLogging, 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 = childLayout->GetMargin();
137
138       auto length = childHeight + LayoutLength::IntType(childMargin.top + childMargin.bottom );
139
140       auto cellPadding = i<GetChildCount()-1 ? mCellPadding.height : 0;
141       auto totalLength = mTotalLength;
142       mTotalLength = std::max( totalLength, totalLength + length + cellPadding);
143
144       bool matchWidthLocally = false;
145       if( widthMode != MeasureSpec::Mode::EXACTLY && desiredWidth == Toolkit::ChildLayoutData::MATCH_PARENT )
146       {
147         // Will have to re-measure at least this child when we know exact height.
148         matchWidth = true;
149         matchWidthLocally = true;
150       }
151
152       auto marginWidth = LayoutLength( childMargin.start + childMargin.end );
153       auto childWidth = childLayout->GetMeasuredWidth() + marginWidth;
154
155       // was combineMeasuredStates()
156       if( childLayout->GetMeasuredWidthAndState().GetState() == MeasuredSize::State::MEASURED_SIZE_TOO_SMALL )
157       {
158         childState.widthState = MeasuredSize::State::MEASURED_SIZE_TOO_SMALL;
159       }
160       if( childLayout->GetMeasuredHeightAndState().GetState() == MeasuredSize::State::MEASURED_SIZE_TOO_SMALL )
161       {
162         childState.heightState = MeasuredSize::State::MEASURED_SIZE_TOO_SMALL;
163       }
164
165       maxWidth = std::max( maxWidth, childWidth );
166       allFillParent = ( allFillParent && desiredWidth == Toolkit::ChildLayoutData::MATCH_PARENT );
167       alternativeMaxWidth = std::max( alternativeMaxWidth, matchWidthLocally ? marginWidth : childWidth );
168     }
169   }
170   Extents padding = GetPadding();
171   mTotalLength += padding.top + padding.bottom;
172   auto heightSize = mTotalLength;
173   heightSize = std::max( heightSize, GetSuggestedMinimumHeight() );
174   MeasuredSize heightSizeAndState = ResolveSizeAndState( heightSize, heightMeasureSpec, MeasuredSize::State::MEASURED_SIZE_OK);
175   heightSize = heightSizeAndState.GetSize();
176
177   if( !allFillParent && widthMode != MeasureSpec::Mode::EXACTLY )
178   {
179     maxWidth = alternativeMaxWidth;
180   }
181   maxWidth += padding.start + padding.end;
182   maxWidth = std::max( maxWidth, GetSuggestedMinimumWidth() );
183
184   heightSizeAndState.SetState( childState.heightState );
185
186   SetMeasuredDimensions( ResolveSizeAndState( maxWidth, widthMeasureSpec, childState.widthState ),
187                          heightSizeAndState );
188
189   if( matchWidth )
190   {
191     ForceUniformWidth( GetChildCount(), heightMeasureSpec );
192   }
193 }
194
195 void VboxLayout::ForceUniformWidth( int count, MeasureSpec heightMeasureSpec )
196 {
197   // Pretend that the linear layout has an exact size. This is the measured height of
198   // ourselves. The measured height should be the max height of the children, changed
199   // to accommodate the heightMeasureSpec from the parent
200   auto uniformMeasureSpec = MeasureSpec( GetMeasuredWidth(), MeasureSpec::Mode::EXACTLY );
201   for (int i = 0; i < count; ++i)
202   {
203     LayoutItemPtr childLayout = GetChildAt(i);
204     if( childLayout != nullptr )
205     {
206       auto childOwner = childLayout->GetOwner();
207       auto desiredWidth = childOwner.GetProperty<int>( Toolkit::LayoutItem::ChildProperty::WIDTH_SPECIFICATION );
208       auto desiredHeight = childOwner.GetProperty<int>( Toolkit::LayoutItem::ChildProperty::HEIGHT_SPECIFICATION );
209
210       if( desiredWidth == Toolkit::ChildLayoutData::MATCH_PARENT )
211       {
212         // Temporarily force children to reuse their old measured height
213         int oldHeight = desiredHeight;
214         childOwner.SetProperty( Toolkit::LayoutItem::ChildProperty::HEIGHT_SPECIFICATION, childLayout->GetMeasuredHeight().mValue );
215
216         // Remeasure with new dimensions
217         MeasureChildWithMargins( childLayout, uniformMeasureSpec, 0, heightMeasureSpec, 0 );
218
219         childOwner.SetProperty( Toolkit::LayoutItem::ChildProperty::HEIGHT_SPECIFICATION, oldHeight );
220       }
221     }
222   }
223 }
224
225 void VboxLayout::OnLayout( bool changed, LayoutLength left, LayoutLength top, LayoutLength right, LayoutLength bottom )
226 {
227   Extents padding = GetPadding();
228
229   LayoutLength childTop( 0 );
230   LayoutLength childLeft( padding.start );
231
232   // Where bottom of child should go
233   auto width = right - left;
234
235   // Space available for child
236   auto childSpace = width - padding.start - padding.end;
237   auto count = GetChildCount();
238
239   for( unsigned int childIndex = 0; childIndex < count; childIndex++)
240   {
241     LayoutItemPtr childLayout = GetChildAt( childIndex );
242     if( childLayout != nullptr )
243     {
244       auto childWidth = childLayout->GetMeasuredWidth();
245       auto childHeight = childLayout->GetMeasuredHeight();
246
247       auto childOwner = childLayout->GetOwner();
248       auto childMargin = childLayout->GetMargin();
249
250       childTop += childMargin.top;
251       childLeft = ( childSpace - childWidth ) / 2 + childMargin.start - childMargin.end;
252
253       childLayout->Layout( childLeft, childTop, childLeft + childWidth, childTop + childHeight );
254       childTop += childHeight + childMargin.bottom + mCellPadding.height;
255     }
256   }
257 }
258
259 } // namespace Internal
260 } // namespace Toolkit
261 } // namespace Dali