7938835e613314872bfd0b101402044893b3d1c0
[platform/core/uifw/dali-core.git] / automated-tests / src / dali / dali-test-suite-utils / test-platform-abstraction.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 #include "test-platform-abstraction.h"
18 #include "dali-test-suite-utils.h"
19 #include <dali/integration-api/bitmap.h>
20
21 namespace Dali
22 {
23
24 /**
25  * Constructor
26  */
27 TestPlatformAbstraction::TestPlatformAbstraction()
28 : mRequest(0)
29 {
30   Initialize();
31 }
32
33 /**
34  * Destructor
35  */
36 TestPlatformAbstraction::~TestPlatformAbstraction()
37 {
38 }
39
40 /**
41  * @copydoc PlatformAbstraction::GetTimeMicroseconds()
42  */
43 void TestPlatformAbstraction::GetTimeMicroseconds(unsigned int &seconds, unsigned int &microSeconds)
44 {
45   seconds = mSeconds;
46   microSeconds = mMicroSeconds;
47   mTrace.PushCall("GetTimeMicroseconds", "");
48 }
49
50 /**
51  * @copydoc PlatformAbstraction::Suspend()
52  */
53 void TestPlatformAbstraction::Suspend()
54 {
55   mTrace.PushCall("Suspend", "");
56 }
57
58 /**
59  * @copydoc PlatformAbstraction::Resume()
60  */
61 void TestPlatformAbstraction::Resume()
62 {
63   mTrace.PushCall("Resume", "");
64 }
65
66 void TestPlatformAbstraction::GetClosestImageSize( const std::string& filename,
67                                                    const ImageAttributes& attributes,
68                                                    Vector2& closestSize)
69 {
70   closestSize = mClosestSize;
71   mTrace.PushCall("GetClosestImageSize", "");
72 }
73
74 void TestPlatformAbstraction::GetClosestImageSize( Integration::ResourcePointer resourceBuffer,
75                                                    const ImageAttributes& attributes,
76                                                    Vector2& closestSize)
77 {
78   closestSize = mClosestSize;
79   mTrace.PushCall("GetClosestImageSize", "");
80 }
81
82
83 /**
84  * @copydoc PlatformAbstraction::LoadResource()
85  */
86 void TestPlatformAbstraction::LoadResource(const Integration::ResourceRequest& request)
87 {
88   mTrace.PushCall("LoadResource", "");
89   if(mRequest != NULL)
90   {
91     delete mRequest;
92     tet_infoline ("Warning: multiple resource requests not handled by Test Suite. You may see unexpected errors");
93   }
94   mRequest = new Integration::ResourceRequest(request);
95 }
96
97 Integration::ResourcePointer TestPlatformAbstraction::LoadResourceSynchronously( const Integration::ResourceType& resourceType, const std::string& resourcePath )
98 {
99   mTrace.PushCall("LoadResourceSynchronously", "");
100   return mResources.loadedResource;
101 }
102
103 /**
104  * @copydoc PlatformAbstraction::SaveResource()
105  */
106 void TestPlatformAbstraction::SaveResource(const Integration::ResourceRequest& request)
107 {
108   mTrace.PushCall("SaveResource", "");
109   if(mRequest != NULL)
110   {
111     delete mRequest;
112     tet_infoline ("Warning: multiple resource requests not handled by Test Suite. You may see unexpected errors");
113   }
114   mRequest = new Integration::ResourceRequest(request);
115 }
116
117 /**
118  * @copydoc PlatformAbstraction::CancelLoad()
119  */
120 void TestPlatformAbstraction::CancelLoad(Integration::ResourceId id, Integration::ResourceTypeId typeId)
121 {
122   mTrace.PushCall("CancelLoad", "");
123 }
124
125 /**
126  * @copydoc PlatformAbstraction::GetResources()
127  */
128 void TestPlatformAbstraction::GetResources(Integration::ResourceCache& cache)
129 {
130   mTrace.PushCall("GetResources", "");
131
132   if(mResources.loaded)
133   {
134     cache.LoadResponse( mResources.loadedId, mResources.loadedType, mResources.loadedResource, Integration::RESOURCE_COMPLETELY_LOADED );
135   }
136   if(mResources.loadFailed)
137   {
138     cache.LoadFailed( mResources.loadFailedId, mResources.loadFailure );
139   }
140   if(mResources.saved)
141   {
142     cache.SaveComplete( mResources.savedId, mResources.savedType );
143   }
144   if(mResources.saveFailed)
145   {
146     cache.SaveFailed( mResources.saveFailedId, mResources.saveFailure );
147   }
148 }
149
150 /**
151  * @copydoc PlatformAbstraction::IsLoading()
152  */
153 bool TestPlatformAbstraction::IsLoading()
154 {
155   mTrace.PushCall("IsLoading", "");
156   return mIsLoadingResult;
157 }
158
159 /**
160  * @copydoc PlatformAbstraction::GetDefaultFontFamily()
161  */
162 std::string TestPlatformAbstraction::GetDefaultFontFamily() const
163 {
164   mTrace.PushCall("GetDefaultFontFamily", "");
165   return mGetDefaultFontFamilyResult;
166 }
167
168 /**
169  * @copydoc PlatformAbstraction::GetDefaultFontSize()
170  */
171 const float TestPlatformAbstraction::GetDefaultFontSize() const
172 {
173   mTrace.PushCall("GetDefaultFontSize", "");
174   return mGetDefaultFontSizeResult;
175 }
176
177 const PixelSize TestPlatformAbstraction::GetFontLineHeightFromCapsHeight(const std::string& fontFamily, const std::string& fontStyle, const CapsHeight& capsHeight) const
178 {
179   mTrace.PushCall("GetFontLineHeightFromCapsHeight", "");
180   // LineHeight will be bigger than CapsHeight, so return capsHeight + 1
181   return PixelSize(capsHeight + 1);
182 }
183
184 /**
185  * @copydoc PlatformAbstraction::GetGlyphData()
186  */
187
188 Integration::GlyphSet* TestPlatformAbstraction::GetGlyphData ( const Integration::TextResourceType& textRequest,
189                                                                const std::string& fontFamily,
190                                                                bool getImageData) const
191 {
192   if( getImageData )
193   {
194     mTrace.PushCall("GetGlyphData", "getImageData:true");
195   }
196   else
197   {
198     mTrace.PushCall("GetGlyphData", "getImageData:false");
199   }
200
201   // It creates fake metrics for the received characters.
202
203   Integration::GlyphSet* set = new Dali::Integration::GlyphSet();
204   Integration::ImageDataPtr imageData;
205
206   std::set<uint32_t> characters;
207
208   for( Integration::TextResourceType::CharacterList::const_iterator it = textRequest.mCharacterList.begin(), endIt = textRequest.mCharacterList.end(); it != endIt; ++it )
209   {
210     if( characters.find( it->character ) == characters.end() )
211     {
212       characters.insert( it->character );
213       Integration::GlyphMetrics character = {it->character, Integration::GlyphMetrics::LOW_QUALITY,  10.0f,  10.0f, 9.0f, 1.0f, 10.0f, it->xPosition, it->yPosition };
214
215       if( getImageData )
216       {
217         // Allocate an ImageData object to hold some pixels:
218         const unsigned dimension = 64;
219         imageData = Integration::NewBitmapImageData( dimension, dimension, Pixel::A8 );
220         uint8_t* const pixelBuffer = imageData->GetBuffer();
221         memset( pixelBuffer, it->character, dimension * dimension );
222       }
223       set->AddCharacter(imageData, character);
224     }
225   }
226
227   set->mLineHeight = 10.0f;
228   set->mAscender = 9.0f;
229   set->mUnitsPerEM = 2048.0f/64.0f;
230   set->SetAtlasResourceId( textRequest.mTextureAtlasId );
231   set->mFontHash = textRequest.mFontHash;
232
233   return set;
234 }
235
236 /**
237  * @copydoc PlatformAbstraction::GetCachedGlyphData()
238  */
239
240 Integration::GlyphSet* TestPlatformAbstraction::GetCachedGlyphData( const Integration::TextResourceType& textRequest,
241                                                                     const std::string& fontFamily ) const
242 {
243   mTrace.PushCall("GetCachedGlyphData", "");
244
245   // It creates fake metrics and bitmap for received numeric characters '0' through '9'.
246   Integration::GlyphSet* set = new Dali::Integration::GlyphSet();
247   Integration::ImageDataPtr imageData;
248
249   std::set<uint32_t> characters;
250
251   for( Integration::TextResourceType::CharacterList::const_iterator it = textRequest.mCharacterList.begin(), endIt = textRequest.mCharacterList.end(); it != endIt; ++it )
252   {
253     if( it->character >= '0' && it->character <= '9' && characters.find( it->character ) == characters.end() )
254     {
255       characters.insert( it->character );
256       Integration::GlyphMetrics character = {it->character, Integration::GlyphMetrics::HIGH_QUALITY,  10.0f,  10.0f, 9.0f, 1.0f, 10.0f, it->xPosition, it->yPosition };
257
258       imageData = Integration::NewBitmapImageData( 64, 64, Pixel::A8 );
259       PixelBuffer* pixelBuffer = imageData->GetBuffer();
260       memset( pixelBuffer, it->character, 64*64 );
261       set->AddCharacter(imageData, character);
262     }
263   }
264
265   set->mLineHeight = 10.0f;
266   set->mAscender = 9.0f;
267   set->mUnitsPerEM = 2048.0f/64.0f;
268   set->SetAtlasResourceId( textRequest.mTextureAtlasId );
269   set->mFontHash = textRequest.mFontHash;
270
271   return set;
272 }
273
274
275 /**
276  * @copydoc PlatformAbstraction::GetGlobalMetrics()
277  */
278 void TestPlatformAbstraction::GetGlobalMetrics( const std::string& fontFamily,
279                                                 const std::string& fontStyle,
280                                                 Integration::GlobalMetrics& globalMetrics ) const
281 {
282   globalMetrics.lineHeight = 10.0f;
283   globalMetrics.ascender = 9.0f;
284   globalMetrics.unitsPerEM = 2048.0f/64.0f;
285   globalMetrics.underlineThickness = 2.f;
286   globalMetrics.underlinePosition = 1.f;
287 }
288
289 /**
290  * @copydoc PlatformAbstraction::GetFontPath()
291  */
292 std::string TestPlatformAbstraction::GetFontPath(const std::string& family, bool bold, bool italic) const
293 {
294   mTrace.PushCall("GetFontPath", "");
295   return mGetFontPathResult;
296
297   // Do nothing with arguments
298 }
299
300 /**
301  * @copydoc PlatformAbstraction::SetDpi()
302  */
303 void TestPlatformAbstraction::SetDpi (unsigned int dpiHorizontal, unsigned int dpiVertical)
304 {
305   mTrace.PushCall("SetDpi", "");
306 }
307
308 /**
309  * @copydoc PlatformAbstraction::GetFontFamilyForChars()
310  */
311 std::string TestPlatformAbstraction::GetFontFamilyForChars(const TextArray& charsRequested) const
312 {
313   mTrace.PushCall("GetFontFamilyForChars", "");
314   return mGetDefaultFontFamilyResult;
315 }
316
317 /**
318  * @copydoc PlatformAbstraction::AllGlyphsSupported()
319  */
320 bool TestPlatformAbstraction::AllGlyphsSupported(const std::string& name, const std::string& fontStyle, const TextArray& text) const
321 {
322   mTrace.PushCall("AllGlyphsSupported", "");
323   return true;
324 }
325
326 /**
327  * @copydoc PlatformAbstraction::ValidateFontFamilyName()
328  */
329 bool TestPlatformAbstraction::ValidateFontFamilyName(const std::string& fontFamily, const std::string& fontStyle, bool& isDefaultSystemFont, std::string& closestMatch, std::string& closestStyleMatch) const
330 {
331   mTrace.PushCall("ValidateFontFamilyName", "");
332   return true;
333 }
334
335 /**
336  * @copydoc PlatformAbstraction::GetFontList()
337  */
338 std::vector<std::string> TestPlatformAbstraction::GetFontList( PlatformAbstraction::FontListMode mode ) const
339 {
340   mFontListMode = mode;
341   mTrace.PushCall("ValidateGetFontList", "");
342   std::vector<std::string> fontList;
343   return fontList;
344 }
345
346 /**
347  * @copydoc PlatformAbstraction::LoadFile()
348  */
349 bool TestPlatformAbstraction::LoadFile( const std::string& filename, std::vector< unsigned char >& buffer ) const
350 {
351   mTrace.PushCall("LoadFile", "");
352   if( mLoadFileResult.loadResult )
353   {
354     buffer = mLoadFileResult.buffer;
355   }
356
357   return mLoadFileResult.loadResult;
358 }
359
360 /**
361  * @copydoc PlatformAbstraction::SaveFile()
362  */
363 bool TestPlatformAbstraction::SaveFile(const std::string& filename, std::vector< unsigned char >& buffer) const
364 {
365   mTrace.PushCall("SaveFile", "");
366   return false;
367 }
368
369 void TestPlatformAbstraction::JoinLoaderThreads()
370 {
371   mTrace.PushCall("JoinLoaderThreads", "");
372 }
373
374 void TestPlatformAbstraction::UpdateDefaultsFromDevice()
375 {
376   mTrace.PushCall("UpdateDefaultsFromDevice", "");
377   mGetDefaultFontFamilyResult+=1.0f;
378 }
379
380 Integration::DynamicsFactory* TestPlatformAbstraction::GetDynamicsFactory()
381 {
382   mTrace.PushCall("GetDynamicsFactory", "");
383   return NULL;
384 }
385
386 bool TestPlatformAbstraction::ReadGlobalMetricsFromCache( const std::string& fontFamily,
387                                                           const std::string& fontStyle,
388                                                           Integration::GlobalMetrics& globalMetrics )
389 {
390   mTrace.PushCall("ReadGlobalMetricsFromCacheFile", "");
391   globalMetrics = mReadGlobalMetrics; // Want to copy contents...
392   return mReadGlobalMetricsResult; // Default false (will be set to true on subsequent write)
393 }
394
395 void TestPlatformAbstraction::WriteGlobalMetricsToCache( const std::string& fontFamily,
396                                                          const std::string& fontStyle,
397                                                          const Integration::GlobalMetrics& globalMetrics )
398 {
399   // Copy so next read uses written values. TODO: Could add method
400   // to turn this behaviour off for more extensive testing.
401   mReadGlobalMetrics = globalMetrics;
402   mReadGlobalMetricsResult = true;
403
404   mTrace.PushCall("WriteGlobalMetricsToCacheFile", "");
405 }
406
407 bool TestPlatformAbstraction::ReadMetricsFromCache( const std::string& fontFamily,
408                                                     const std::string& fontStyle,
409                                                     std::vector<Integration::GlyphMetrics>& glyphMetricsContainer )
410 {
411   mTrace.PushCall("ReadMetricsFromCacheFile", "");
412   glyphMetricsContainer = mReadMetrics;
413   return mReadMetricsResult; // Default false (will be set to true on subsequent write)
414 }
415
416 void TestPlatformAbstraction::WriteMetricsToCache( const std::string& fontFamily,
417                                                    const std::string& fontStyle,
418                                                    const Integration::GlyphSet& glyphSet )
419 {
420   // Copy so next read uses written values. TODO: Could add method
421   // to turn this behaviour off for more extensive testing.
422   const Integration::GlyphSet::CharacterList& charList =  glyphSet.GetCharacterList();
423   mReadMetrics.clear();
424   for(std::size_t i=0, end=charList.size(); i<end; ++i)
425   {
426     mReadMetrics.push_back(charList[i].second);
427   }
428   mReadMetricsResult = true;
429
430   mTrace.PushCall("WriteMetricsToCacheFile", "");
431 }
432
433
434 void TestPlatformAbstraction::GetFileNamesFromDirectory( const std::string& directoryName,
435                                                          std::vector<std::string>& fileNames )
436 {
437   fileNames.push_back( std::string( "u1f004.png" ) );
438   fileNames.push_back( std::string( "u1f0cf.png" ) );
439   fileNames.push_back( std::string( "u1f170.png" ) );
440   fileNames.push_back( std::string( "u1f601.png" ) );
441 }
442
443
444 Integration::ImageDataPtr TestPlatformAbstraction::GetGlyphImage( const std::string& fontFamily, const std::string& fontStyle, float fontSize, uint32_t character ) const
445 {
446   Integration::ImageDataPtr image = Integration::NewBitmapImageData( 1, 1, Pixel::RGBA8888 );
447
448   mTrace.PushCall("GetGlyphImage", "");
449
450   return image;
451 }
452
453
454 /** Call this every test */
455 void TestPlatformAbstraction::Initialize()
456 {
457   mTrace.Reset();
458   mTrace.Enable(true);
459   memset(&mResources, 0, sizeof(Resources));
460   memset(&mReadGlobalMetrics, 0, sizeof(Integration::GlobalMetrics));
461   mSeconds=0;
462   mMicroSeconds=0;
463   mIsLoadingResult=false;
464   mGetDefaultFontFamilyResult = "HelveticaNeue";
465   mGetDefaultFontSizeResult=12.0f;
466   mGetFontPathResult="helvetica-12";
467   mReadMetricsResult=false;
468   mReadGlobalMetricsResult=false;
469
470   if(mRequest)
471   {
472     delete mRequest;
473     mRequest = 0;
474   }
475 }
476
477
478 bool TestPlatformAbstraction::WasCalled(TestFuncEnum func)
479 {
480   switch(func)
481   {
482     case GetTimeMicrosecondsFunc:             return mTrace.FindMethod("GetTimeMicroseconds");
483     case SuspendFunc:                         return mTrace.FindMethod("Suspend");
484     case ResumeFunc:                          return mTrace.FindMethod("Resume");
485     case LoadResourceFunc:                    return mTrace.FindMethod("LoadResource");
486     case SaveResourceFunc:                    return mTrace.FindMethod("SaveResource");
487     case LoadFileFunc:                        return mTrace.FindMethod("LoadFile");
488     case SaveFileFunc:                        return mTrace.FindMethod("SaveFile");
489     case CancelLoadFunc:                      return mTrace.FindMethod("CancelLoad");
490     case GetResourcesFunc:                    return mTrace.FindMethod("GetResources");
491     case IsLoadingFunc:                       return mTrace.FindMethod("IsLoading");
492     case GetDefaultFontFamilyFunc:            return mTrace.FindMethod("GetDefaultFontFamily");
493     case GetDefaultFontSizeFunc:              return mTrace.FindMethod("GetDefaultFontSize");
494     case GetFontLineHeightFromCapsHeightFunc: return mTrace.FindMethod("GetFontLineHeightFromCapsHeight");
495     case GetGlyphDataFunc:                    return mTrace.FindMethod("GetGlyphData");
496     case GetCachedGlyphDataFunc:              return mTrace.FindMethod("GetCachedGlyphData");
497     case GetFontPathFunc:                     return mTrace.FindMethod("GetFontPath");
498     case SetDpiFunc:                          return mTrace.FindMethod("SetDpi");
499     case JoinLoaderThreadsFunc:               return mTrace.FindMethod("JoinLoaderThreads");
500     case GetFontFamilyForCharsFunc:           return mTrace.FindMethod("GetFontFamilyForChars");
501     case AllGlyphsSupportedFunc:              return mTrace.FindMethod("AllGlyphsSupported");
502     case ValidateFontFamilyNameFunc:          return mTrace.FindMethod("ValidateFontFamilyName");
503     case UpdateDefaultsFromDeviceFunc:        return mTrace.FindMethod("UpdateDefaultsFromDevice");
504     case GetDynamicsFactoryFunc:              return mTrace.FindMethod("GetDynamicsFactory");
505     case ValidateGetFontListFunc:             return mTrace.FindMethod("ValidateGetFontList");
506     case ReadGlobalMetricsFromCacheFileFunc:  return mTrace.FindMethod("ReadGlobalMetricsFromCacheFile");
507     case WriteGlobalMetricsToCacheFileFunc:   return mTrace.FindMethod("WriteGlobalMetricsToCacheFile");
508     case ReadMetricsFromCacheFileFunc:        return mTrace.FindMethod("ReadMetricsFromCacheFile");
509     case WriteMetricsToCacheFileFunc:         return mTrace.FindMethod("WriteMetricsToCacheFile");
510   }
511   return false;
512 }
513
514 void TestPlatformAbstraction::SetGetTimeMicrosecondsResult(size_t sec, size_t usec)
515 {
516   mSeconds = sec;
517   mMicroSeconds = usec;
518 }
519
520 void TestPlatformAbstraction::IncrementGetTimeResult(size_t milliseconds)
521 {
522   mMicroSeconds += milliseconds * 1000u;
523   unsigned int additionalSeconds = mMicroSeconds / 1000000u;
524
525   mSeconds += additionalSeconds;
526   mMicroSeconds -= additionalSeconds * 1000000u;
527 }
528
529 void TestPlatformAbstraction::SetIsLoadingResult(bool result)
530 {
531   mIsLoadingResult = result;
532 }
533
534 void TestPlatformAbstraction::SetGetDefaultFontFamilyResult(std::string result)
535 {
536   mGetDefaultFontFamilyResult = result;
537 }
538
539 void TestPlatformAbstraction::SetGetDefaultFontSizeResult(float result)
540 {
541   mGetDefaultFontSizeResult = result;
542 }
543
544 void TestPlatformAbstraction::SetGetFontPathResult(std::string& result)
545 {
546   mGetFontPathResult = result;
547 }
548
549 void TestPlatformAbstraction::ClearReadyResources()
550 {
551   memset(&mResources, 0, sizeof(Resources));
552 }
553
554 void TestPlatformAbstraction::SetResourceLoaded(Integration::ResourceId  loadedId,
555                                                 Integration::ResourceTypeId  loadedType,
556                                                 Integration::ResourcePointer loadedResource)
557 {
558   mResources.loaded = true;
559   mResources.loadedId = loadedId;
560   mResources.loadedType = loadedType;
561   mResources.loadedResource = loadedResource;
562 }
563
564 void TestPlatformAbstraction::SetResourceLoadFailed(Integration::ResourceId  id,
565                                                     Integration::ResourceFailure failure)
566 {
567   mResources.loadFailed = true;
568   mResources.loadFailedId = id;
569   mResources.loadFailure = failure;
570 }
571
572 void TestPlatformAbstraction::SetResourceSaved(Integration::ResourceId      savedId,
573                                                Integration::ResourceTypeId  savedType)
574 {
575   mResources.saved = true;
576   mResources.savedId = savedId;
577   mResources.savedType = savedType;
578 }
579
580 void TestPlatformAbstraction::SetResourceSaveFailed(Integration::ResourceId  id,
581                                                     Integration::ResourceFailure failure)
582 {
583   mResources.saveFailed = true;
584   mResources.saveFailedId = id;
585   mResources.saveFailure = failure;
586 }
587
588 Integration::ResourceRequest* TestPlatformAbstraction::GetRequest()
589 {
590   return mRequest;
591 }
592
593 void TestPlatformAbstraction::DiscardRequest()
594 {
595   delete mRequest;
596   mRequest = NULL;
597 }
598
599 void TestPlatformAbstraction::SetClosestImageSize(const Vector2& size)
600 {
601   mClosestSize = size;
602 }
603
604 void TestPlatformAbstraction::SetLoadFileResult( bool result, std::vector< unsigned char >& buffer )
605 {
606   mLoadFileResult.loadResult = result;
607   if( result )
608   {
609     mLoadFileResult.buffer = buffer;
610   }
611 }
612
613 void TestPlatformAbstraction::SetSaveFileResult( bool result )
614 {
615   mSaveFileResult = result;
616 }
617
618 Integration::PlatformAbstraction::FontListMode TestPlatformAbstraction::GetLastFontListMode( )
619 {
620   return mFontListMode;
621 }
622
623 void TestPlatformAbstraction::SetReadGlobalMetricsResult( bool success, Integration::GlobalMetrics& globalMetrics )
624 {
625   mReadGlobalMetricsResult = success;
626   mReadGlobalMetrics = globalMetrics;
627 }
628
629 void TestPlatformAbstraction::SetReadMetricsResult( bool success, std::vector<Integration::GlyphMetrics>& glyphMetricsContainer )
630 {
631   mReadMetricsResult = success;
632   mReadMetrics = glyphMetricsContainer; // copy
633 }
634
635 } // namespace Dali