[Tizen] Add support for FontClientFontPreLoad API
[platform/core/uifw/dali-adaptor.git] / dali / internal / text / text-abstraction / font-client-impl.cpp
old mode 100755 (executable)
new mode 100644 (file)
index 4e82a54..1a0bd45
@@ -1,5 +1,5 @@
 /*
- * Copyright (c) 2019 Samsung Electronics Co., Ltd.
+ * Copyright (c) 2022 Samsung Electronics Co., Ltd.
  *
  * Licensed under the Apache License, Version 2.0 (the "License");
  * you may not use this file except in compliance with the License.
 #include <dali/internal/text/text-abstraction/font-client-impl.h>
 
 // EXTERNAL INCLUDES
-#if !(defined(DALI_PROFILE_UBUNTU) || defined(ANDROID) || defined(WIN32))
+#include <mutex>
+#include <thread>
+#if defined(VCONF_ENABLED)
 #include <vconf.h>
 #endif
 
 // INTERNAL INCLUDES
 #include <dali/devel-api/common/singleton-service.h>
-#include <dali/internal/text/text-abstraction/font-client-plugin-impl.h>
+#include <dali/internal/system/common/logging.h>
+#include <dali/internal/text/text-abstraction/plugin/font-client-plugin-impl.h>
+#include <dali/internal/window-system/common/window-system.h>
 
 #include <dali/devel-api/text-abstraction/glyph-info.h>
 
+// Use this macro only if need to log messages before the log function is set.
+#define FONT_LOG_MESSAGE(level, format, ...)                                    \
+  do                                                                            \
+  {                                                                             \
+    char buffer[256];                                                           \
+    int  result = std::snprintf(buffer, sizeof(buffer), format, ##__VA_ARGS__); \
+    if(result >= static_cast<int>(sizeof(buffer)))                              \
+    {                                                                           \
+      std::string log("Font log message is too long to fit in the buffer.\n");  \
+      Dali::TizenPlatform::LogMessage(Dali::Integration::Log::ERROR, log);      \
+      break;                                                                    \
+    }                                                                           \
+    std::string log(buffer);                                                    \
+    Dali::TizenPlatform::LogMessage(level, log);                                \
+  } while(0)
+
 namespace Dali
 {
-
 namespace TextAbstraction
 {
-
 namespace Internal
 {
+Dali::TextAbstraction::FontClient FontClient::gPreCreatedFontClient(NULL);
+std::thread                       gPreCacheThread;
+std::thread                       gPreLoadThread;
+std::mutex                        gMutex;
 
-Dali::TextAbstraction::FontClient FontClient::gPreInitializedFontClient( NULL );
+/* TODO: This is to prevent duplicate calls of font pre-cache.
+ * We may support this later, but currently we can't guarantee the behaviour
+ * if there is a pre-cache call from the user after the font client has been created. */
+bool gFontPreCacheAvailable = true;
+bool gFontPreLoadAvailable  = true;
 
 FontClient::FontClient()
-: mPlugin( nullptr ),
-  mDpiHorizontal( 0 ),
-  mDpiVertical( 0 )
+: mPlugin(nullptr),
+  mDpiHorizontal(0),
+  mDpiVertical(0)
 {
 }
 
@@ -56,30 +82,56 @@ Dali::TextAbstraction::FontClient FontClient::Get()
 {
   Dali::TextAbstraction::FontClient fontClientHandle;
 
-  Dali::SingletonService service( SingletonService::Get() );
-  if ( service )
+  Dali::SingletonService service(SingletonService::Get());
+  if(service)
   {
     // Check whether the singleton is already created
-    Dali::BaseHandle handle = service.GetSingleton( typeid( Dali::TextAbstraction::FontClient ) );
+    Dali::BaseHandle handle = service.GetSingleton(typeid(Dali::TextAbstraction::FontClient));
     if(handle)
     {
       // If so, downcast the handle
-      FontClient* impl = dynamic_cast< Dali::TextAbstraction::Internal::FontClient* >( handle.GetObjectPtr() );
-      fontClientHandle = Dali::TextAbstraction::FontClient( impl );
+      FontClient* impl = dynamic_cast<Dali::TextAbstraction::Internal::FontClient*>(handle.GetObjectPtr());
+      fontClientHandle = Dali::TextAbstraction::FontClient(impl);
     }
     else // create and register the object
     {
-      if( gPreInitializedFontClient )
+      if(gPreCacheThread.joinable())
+      {
+        gPreCacheThread.join();
+        FONT_LOG_MESSAGE(Dali::Integration::Log::INFO, "FontClient PreCache thread join\n");
+      }
+
+      if(gPreLoadThread.joinable())
+      {
+        gPreLoadThread.join();
+        FONT_LOG_MESSAGE(Dali::Integration::Log::INFO, "FontClient PreLoad thread join\n");
+      }
+
+      if(gPreCreatedFontClient)
       {
-        fontClientHandle = gPreInitializedFontClient;
-        gPreInitializedFontClient.Reset(); // No longer needed
+        fontClientHandle = gPreCreatedFontClient;
+        gPreCreatedFontClient.Reset(); // No longer needed
       }
       else
       {
-        fontClientHandle = Dali::TextAbstraction::FontClient( new FontClient );
+        fontClientHandle = Dali::TextAbstraction::FontClient(new FontClient);
+      }
+
+      fontClientHandle.InitDefaultFontDescription();
+
+      gFontPreCacheAvailable = false;
+      gFontPreLoadAvailable  = false;
+
+      uint32_t horizontalDpi, verticalDpi;
+      fontClientHandle.GetDpi(horizontalDpi, verticalDpi);
+      if(horizontalDpi == 0u || verticalDpi == 0u)
+      {
+        horizontalDpi = verticalDpi = 0u;
+        Dali::Internal::Adaptor::WindowSystem::GetDpi(horizontalDpi, verticalDpi);
+        fontClientHandle.SetDpi(horizontalDpi, verticalDpi);
       }
 
-      service.Register( typeid( fontClientHandle ), fontClientHandle );
+      service.Register(typeid(fontClientHandle), fontClientHandle);
     }
   }
 
@@ -88,48 +140,139 @@ Dali::TextAbstraction::FontClient FontClient::Get()
 
 Dali::TextAbstraction::FontClient FontClient::PreInitialize()
 {
-  gPreInitializedFontClient = Dali::TextAbstraction::FontClient( new FontClient );
+  // Pre-cached font client already exists or pre-cache thread already running.
+  // Font client pre-cache includes caching of the default font description.
+  if(gPreCreatedFontClient && !gFontPreCacheAvailable)
+  {
+    return gPreCreatedFontClient;
+  }
 
-  // Make DefaultFontDescription cached
-  Dali::TextAbstraction::FontDescription defaultFontDescription;
-  gPreInitializedFontClient.GetDefaultPlatformFontDescription( defaultFontDescription );
+  if(!gPreCreatedFontClient)
+  {
+    gPreCreatedFontClient = Dali::TextAbstraction::FontClient(new FontClient);
+  }
+
+  gPreCreatedFontClient.InitDefaultFontDescription();
+
+  return gPreCreatedFontClient;
+}
+
+void FontClient::PreCacheRun(const FontFamilyList& fallbackFamilyList, const FontFamilyList& extraFamilyList, const FontFamily& localeFamily)
+{
+  FONT_LOG_MESSAGE(Dali::Integration::Log::INFO, "BEGIN: DALI_TEXT_PRECACHE_RUN\n");
+  GetImplementation(gPreCreatedFontClient).FontPreCache(fallbackFamilyList, extraFamilyList, localeFamily);
+  FONT_LOG_MESSAGE(Dali::Integration::Log::INFO, "END: DALI_TEXT_PRECACHE_RUN\n");
+}
+
+void FontClient::PreLoadRun(const FontPathList& fontPathList, const FontPathList& memoryFontPathList)
+{
+  FONT_LOG_MESSAGE(Dali::Integration::Log::INFO, "BEGIN: DALI_TEXT_FONT_PRELOAD_RUN\n");
+  GetImplementation(gPreCreatedFontClient).FontPreLoad(fontPathList, memoryFontPathList);
+  FONT_LOG_MESSAGE(Dali::Integration::Log::INFO, "END: DALI_TEXT_FONT_PRELOAD_RUN\n");
+}
+
+void FontClient::PreCache(const FontFamilyList& fallbackFamilyList, const FontFamilyList& extraFamilyList, const FontFamily& localeFamily, bool useThread)
+{
+  if(!gFontPreCacheAvailable)
+  {
+    FONT_LOG_MESSAGE(Dali::Integration::Log::ERROR, "FontClient pre-cache run failed, as a pre-cached font client already exists.\n");
+    return;
+  }
 
-  return gPreInitializedFontClient;
+  if(gPreCacheThread.joinable())
+  {
+    FONT_LOG_MESSAGE(Dali::Integration::Log::ERROR, "FontClient pre-cache thread already running.\n");
+  }
+
+  if(!gPreCreatedFontClient)
+  {
+    gPreCreatedFontClient = Dali::TextAbstraction::FontClient(new FontClient);
+  }
+
+  gFontPreCacheAvailable = false;
+
+  FONT_LOG_MESSAGE(Dali::Integration::Log::INFO, "FontClient FontConfig PreCache fallbackFamilyList : %zu\n", fallbackFamilyList.size());
+  FONT_LOG_MESSAGE(Dali::Integration::Log::INFO, "FontClient FontConfig PreCache extraFamilyList    : %zu\n", extraFamilyList.size());
+  FONT_LOG_MESSAGE(Dali::Integration::Log::INFO, "FontClient FontConfig PreCache localeFamily       : %s\n", localeFamily.c_str());
+  FONT_LOG_MESSAGE(Dali::Integration::Log::INFO, "FontClient FontConfig PreCache useThread          : %d\n", useThread);
+
+  if(useThread)
+  {
+    gPreCacheThread = std::thread(PreCacheRun, fallbackFamilyList, extraFamilyList, localeFamily);
+  }
+  else
+  {
+    PreCacheRun(fallbackFamilyList, extraFamilyList, localeFamily);
+  }
+}
+
+void FontClient::PreLoad(const FontPathList& fontPathList, const FontPathList& memoryFontPathList, bool useThread)
+{
+  if(!gFontPreLoadAvailable)
+  {
+    FONT_LOG_MESSAGE(Dali::Integration::Log::ERROR, "FontClient font pre-load run failed, as a pre-loaded font client already exists.\n");
+    return;
+  }
+
+  if(gPreLoadThread.joinable())
+  {
+    FONT_LOG_MESSAGE(Dali::Integration::Log::ERROR, "FontClient font pre-load thread already running.\n");
+    return;
+  }
+
+  if(!gPreCreatedFontClient)
+  {
+    gPreCreatedFontClient = Dali::TextAbstraction::FontClient(new FontClient);
+  }
+
+  gFontPreLoadAvailable = false;
+
+  FONT_LOG_MESSAGE(Dali::Integration::Log::INFO, "FontClient Font PreLoad fontPathList       : %zu\n", fontPathList.size());
+  FONT_LOG_MESSAGE(Dali::Integration::Log::INFO, "FontClient Font PreLoad memoryFontPathList : %zu\n", memoryFontPathList.size());
+  FONT_LOG_MESSAGE(Dali::Integration::Log::INFO, "FontClient Font PreLoad useThread          : %d\n", useThread);
+
+  if(useThread)
+  {
+    gPreLoadThread = std::thread(PreLoadRun, fontPathList, memoryFontPathList);
+  }
+  else
+  {
+    PreLoadRun(fontPathList, memoryFontPathList);
+  }
 }
 
 void FontClient::ClearCache()
 {
-  if( mPlugin )
+  if(mPlugin)
   {
     mPlugin->ClearCache();
   }
 }
 
-
-void FontClient::SetDpi( unsigned int horizontalDpi, unsigned int verticalDpi  )
+void FontClient::SetDpi(unsigned int horizontalDpi, unsigned int verticalDpi)
 {
   mDpiHorizontal = horizontalDpi;
-  mDpiVertical = verticalDpi;
+  mDpiVertical   = verticalDpi;
 
   // Allow DPI to be set without loading plugin
-  if( mPlugin )
+  if(mPlugin)
   {
-    mPlugin->SetDpi( horizontalDpi, verticalDpi  );
+    mPlugin->SetDpi(horizontalDpi, verticalDpi);
   }
 }
 
-void FontClient::GetDpi( unsigned int& horizontalDpi, unsigned int& verticalDpi )
+void FontClient::GetDpi(unsigned int& horizontalDpi, unsigned int& verticalDpi)
 {
   horizontalDpi = mDpiHorizontal;
-  verticalDpi = mDpiVertical;
+  verticalDpi   = mDpiVertical;
 }
 
 int FontClient::GetDefaultFontSize()
 {
-  int fontSize( -1 );
+  int fontSize(-1);
 
-#if !(defined(DALI_PROFILE_UBUNTU) || defined(ANDROID) || defined(WIN32))
-  vconf_get_int( VCONFKEY_SETAPPL_ACCESSIBILITY_FONT_SIZE, &fontSize );
+#if defined(VCONF_ENABLED)
+  vconf_get_int(VCONFKEY_SETAPPL_ACCESSIBILITY_FONT_SIZE, &fontSize);
 #endif
 
   return fontSize;
@@ -142,227 +285,321 @@ void FontClient::ResetSystemDefaults()
   mPlugin->ResetSystemDefaults();
 }
 
-void FontClient::GetDefaultFonts( FontList& defaultFonts )
+void FontClient::GetDefaultFonts(FontList& defaultFonts)
 {
   CreatePlugin();
 
-  mPlugin->GetDefaultFonts( defaultFonts );
+  mPlugin->GetDefaultFonts(defaultFonts);
 }
 
-void FontClient::GetDefaultPlatformFontDescription( FontDescription& fontDescription )
+void FontClient::FontPreCache(const FontFamilyList& fallbackFamilyList, const FontFamilyList& extraFamilyList, const FontFamily& localeFamily)
 {
   CreatePlugin();
 
-  mPlugin->GetDefaultPlatformFontDescription( fontDescription );
+  mPlugin->FontPreCache(fallbackFamilyList, extraFamilyList, localeFamily);
 }
 
-void FontClient::GetDescription( FontId id, FontDescription& fontDescription )
+void FontClient::FontPreLoad(const FontPathList& fontPathList, const FontPathList& memoryFontPathList)
 {
   CreatePlugin();
 
-  mPlugin->GetDescription( id, fontDescription );
+  mPlugin->FontPreLoad(fontPathList, memoryFontPathList);
 }
 
-PointSize26Dot6 FontClient::GetPointSize( FontId id )
+void FontClient::InitDefaultFontDescription()
 {
   CreatePlugin();
 
-  return mPlugin->GetPointSize( id );
+  mPlugin->InitDefaultFontDescription();
 }
 
-bool FontClient::IsCharacterSupportedByFont( FontId fontId, Character character )
+void FontClient::GetDefaultPlatformFontDescription(FontDescription& fontDescription)
 {
   CreatePlugin();
 
-  return mPlugin->IsCharacterSupportedByFont( fontId, character );
+  mPlugin->GetDefaultPlatformFontDescription(fontDescription);
 }
 
-void FontClient::GetSystemFonts( FontList& systemFonts )
+void FontClient::GetDescription(FontId fontId, FontDescription& fontDescription)
 {
   CreatePlugin();
 
-  mPlugin->GetSystemFonts( systemFonts );
+  mPlugin->GetDescription(fontId, fontDescription);
 }
 
-FontId FontClient::FindDefaultFont( Character charcode,
-                                    PointSize26Dot6 requestedPointSize,
-                                    bool preferColor )
+PointSize26Dot6 FontClient::GetPointSize(FontId fontId)
 {
   CreatePlugin();
 
-  return mPlugin->FindDefaultFont( charcode,
-                                   requestedPointSize,
-                                   preferColor );
+  return mPlugin->GetPointSize(fontId);
+}
+
+bool FontClient::IsCharacterSupportedByFont(FontId fontId, Character character)
+{
+  CreatePlugin();
+
+  return mPlugin->IsCharacterSupportedByFont(fontId, character);
+}
+
+void FontClient::GetSystemFonts(FontList& systemFonts)
+{
+  CreatePlugin();
+
+  mPlugin->GetSystemFonts(systemFonts);
 }
 
-FontId FontClient::FindFallbackFont( Character charcode,
-                                     const FontDescription& preferredFontDescription,
-                                     PointSize26Dot6 requestedPointSize,
-                                     bool preferColor )
+FontId FontClient::FindDefaultFont(Character       charcode,
+                                   PointSize26Dot6 requestedPointSize,
+                                   bool            preferColor)
 {
   CreatePlugin();
 
-  return mPlugin->FindFallbackFont( charcode,
-                                    preferredFontDescription,
-                                    requestedPointSize,
-                                    preferColor );
+  return mPlugin->FindDefaultFont(charcode,
+                                  requestedPointSize,
+                                  preferColor);
 }
 
-bool FontClient::IsScalable( const FontPath& path )
+FontId FontClient::FindFallbackFont(Character              charcode,
+                                    const FontDescription& preferredFontDescription,
+                                    PointSize26Dot6        requestedPointSize,
+                                    bool                   preferColor)
 {
   CreatePlugin();
 
-  return mPlugin->IsScalable( path );
+  return mPlugin->FindFallbackFont(charcode,
+                                   preferredFontDescription,
+                                   requestedPointSize,
+                                   preferColor);
 }
 
-bool FontClient::IsScalable( const FontDescription& fontDescription )
+bool FontClient::IsScalable(const FontPath& path)
 {
   CreatePlugin();
 
-  return mPlugin->IsScalable( fontDescription );
+  return mPlugin->IsScalable(path);
 }
 
-void FontClient::GetFixedSizes( const FontPath& path, Dali::Vector< PointSize26Dot6>& sizes )
+bool FontClient::IsScalable(const FontDescription& fontDescription)
 {
   CreatePlugin();
 
-  mPlugin->GetFixedSizes( path, sizes );
+  return mPlugin->IsScalable(fontDescription);
 }
 
-void FontClient::GetFixedSizes( const FontDescription& fontDescription,
-                                Dali::Vector< PointSize26Dot6 >& sizes )
+void FontClient::GetFixedSizes(const FontPath& path, Dali::Vector<PointSize26Dot6>& sizes)
 {
   CreatePlugin();
 
-  mPlugin->GetFixedSizes( fontDescription, sizes );
+  mPlugin->GetFixedSizes(path, sizes);
 }
 
-bool FontClient::HasItalicStyle( FontId fontId ) const
+void FontClient::GetFixedSizes(const FontDescription&         fontDescription,
+                               Dali::Vector<PointSize26Dot6>& sizes)
 {
-  if( !mPlugin )
+  CreatePlugin();
+
+  mPlugin->GetFixedSizes(fontDescription, sizes);
+}
+
+bool FontClient::HasItalicStyle(FontId fontId) const
+{
+  if(!mPlugin)
   {
     return false;
   }
-  return mPlugin->HasItalicStyle( fontId );
+  return mPlugin->HasItalicStyle(fontId);
+}
+
+FontId FontClient::GetFontId(const FontPath& path, PointSize26Dot6 requestedPointSize, FaceIndex faceIndex)
+{
+  CreatePlugin();
+
+  return mPlugin->GetFontIdByPath(path,
+                                  requestedPointSize,
+                                  faceIndex,
+                                  true);
 }
 
-FontId FontClient::GetFontId( const FontPath& path, PointSize26Dot6 requestedPointSize, FaceIndex faceIndex )
+FontId FontClient::GetFontId(const FontDescription& fontDescription,
+                             PointSize26Dot6        requestedPointSize,
+                             FaceIndex              faceIndex)
 {
   CreatePlugin();
 
-  return mPlugin->GetFontId( path,
-                             requestedPointSize,
-                             faceIndex,
-                             true );
+  return mPlugin->GetFontId(fontDescription,
+                            requestedPointSize,
+                            faceIndex);
 }
 
-FontId FontClient::GetFontId( const FontDescription& fontDescription,
-                              PointSize26Dot6 requestedPointSize,
-                              FaceIndex faceIndex )
+FontId FontClient::GetFontId(const BitmapFont& bitmapFont)
 {
   CreatePlugin();
 
-  return mPlugin->GetFontId( fontDescription,
-                             requestedPointSize,
-                             faceIndex );
+  return mPlugin->GetFontId(bitmapFont);
 }
 
-FontId FontClient::GetFontId( const BitmapFont& bitmapFont )
+void FontClient::GetFontMetrics(FontId fontId, FontMetrics& metrics)
 {
   CreatePlugin();
 
-  return mPlugin->GetFontId( bitmapFont );
+  mPlugin->GetFontMetrics(fontId, metrics);
 }
 
-void FontClient::GetFontMetrics( FontId fontId, FontMetrics& metrics )
+GlyphIndex FontClient::GetGlyphIndex(FontId fontId, Character charcode)
 {
   CreatePlugin();
 
-  mPlugin->GetFontMetrics( fontId, metrics );
+  return mPlugin->GetGlyphIndex(fontId, charcode);
 }
 
-GlyphIndex FontClient::GetGlyphIndex( FontId fontId, Character charcode )
+GlyphIndex FontClient::GetGlyphIndex(FontId fontId, Character charcode, Character variantSelector)
 {
   CreatePlugin();
 
-  return mPlugin->GetGlyphIndex( fontId, charcode );
+  return mPlugin->GetGlyphIndex(fontId, charcode, variantSelector);
 }
 
-bool FontClient::GetGlyphMetrics( GlyphInfo* array, uint32_t size, GlyphType type, bool horizontal )
+bool FontClient::GetGlyphMetrics(GlyphInfo* array, uint32_t size, GlyphType type, bool horizontal)
 {
   CreatePlugin();
 
-  return mPlugin->GetGlyphMetrics( array, size, type, horizontal );
+  return mPlugin->GetGlyphMetrics(array, size, type, horizontal);
 }
 
-void FontClient::CreateBitmap( FontId fontId, GlyphIndex glyphIndex, bool isItalicRequired, bool isBoldRequired, Dali::TextAbstraction::FontClient::GlyphBufferData& data, int outlineWidth )
+void FontClient::CreateBitmap(FontId fontId, GlyphIndex glyphIndex, bool isItalicRequired, bool isBoldRequired, Dali::TextAbstraction::FontClient::GlyphBufferData& data, int outlineWidth)
 {
   CreatePlugin();
 
-  mPlugin->CreateBitmap( fontId, glyphIndex, isItalicRequired, isBoldRequired, data, outlineWidth );
+  mPlugin->CreateBitmap(fontId, glyphIndex, isItalicRequired, isBoldRequired, data, outlineWidth);
 }
 
-PixelData FontClient::CreateBitmap( FontId fontId, GlyphIndex glyphIndex, int outlineWidth )
+PixelData FontClient::CreateBitmap(FontId fontId, GlyphIndex glyphIndex, int outlineWidth)
 {
   CreatePlugin();
 
-  return mPlugin->CreateBitmap( fontId, glyphIndex, outlineWidth );
+  return mPlugin->CreateBitmap(fontId, glyphIndex, outlineWidth);
 }
 
-void FontClient::CreateVectorBlob( FontId fontId, GlyphIndex glyphIndex, VectorBlob*& blob, unsigned int& blobLength, unsigned int& nominalWidth, unsigned int& nominalHeight )
+void FontClient::CreateVectorBlob(FontId fontId, GlyphIndex glyphIndex, VectorBlob*& blob, unsigned int& blobLength, unsigned int& nominalWidth, unsigned int& nominalHeight)
 {
   CreatePlugin();
 
-  mPlugin->CreateVectorBlob( fontId, glyphIndex, blob, blobLength, nominalWidth, nominalHeight );
+  mPlugin->CreateVectorBlob(fontId, glyphIndex, blob, blobLength, nominalWidth, nominalHeight);
 }
 
-const GlyphInfo& FontClient::GetEllipsisGlyph( PointSize26Dot6 requestedPointSize )
+const GlyphInfo& FontClient::GetEllipsisGlyph(PointSize26Dot6 requestedPointSize)
 {
   CreatePlugin();
 
-  return mPlugin->GetEllipsisGlyph( requestedPointSize );
+  return mPlugin->GetEllipsisGlyph(requestedPointSize);
 }
 
-bool FontClient::IsColorGlyph( FontId fontId, GlyphIndex glyphIndex )
+bool FontClient::IsColorGlyph(FontId fontId, GlyphIndex glyphIndex)
 {
   CreatePlugin();
 
-  return mPlugin->IsColorGlyph( fontId, glyphIndex );
+  return mPlugin->IsColorGlyph(fontId, glyphIndex);
 }
 
 GlyphIndex FontClient::CreateEmbeddedItem(const TextAbstraction::FontClient::EmbeddedItemDescription& description, Pixel::Format& pixelFormat)
 {
   CreatePlugin();
 
-  return mPlugin->CreateEmbeddedItem( description, pixelFormat );
+  return mPlugin->CreateEmbeddedItem(description, pixelFormat);
+}
+
+void FontClient::EnableAtlasLimitation(bool enabled)
+{
+  CreatePlugin();
+  return mPlugin->EnableAtlasLimitation(enabled);
+}
+
+bool FontClient::IsAtlasLimitationEnabled() const
+{
+  if(mPlugin)
+  {
+    return mPlugin->IsAtlasLimitationEnabled();
+  }
+  return TextAbstraction::FontClient::DEFAULT_ATLAS_LIMITATION_ENABLED;
+}
+
+Size FontClient::GetMaximumTextAtlasSize() const
+{
+  if(mPlugin)
+  {
+    return mPlugin->GetMaximumTextAtlasSize();
+  }
+  return TextAbstraction::FontClient::MAX_TEXT_ATLAS_SIZE;
+}
+
+Size FontClient::GetDefaultTextAtlasSize() const
+{
+  if(mPlugin)
+  {
+    return mPlugin->GetDefaultTextAtlasSize();
+  }
+  return TextAbstraction::FontClient::DEFAULT_TEXT_ATLAS_SIZE;
+}
+
+Size FontClient::GetCurrentMaximumBlockSizeFitInAtlas() const
+{
+  if(mPlugin)
+  {
+    return mPlugin->GetCurrentMaximumBlockSizeFitInAtlas();
+  }
+  return TextAbstraction::FontClient::DEFAULT_TEXT_ATLAS_SIZE;
+}
+
+bool FontClient::SetCurrentMaximumBlockSizeFitInAtlas(const Size& currentMaximumBlockSizeFitInAtlas)
+{
+  CreatePlugin();
+  return mPlugin->SetCurrentMaximumBlockSizeFitInAtlas(currentMaximumBlockSizeFitInAtlas);
+}
+
+uint32_t FontClient::GetNumberOfPointsPerOneUnitOfPointSize() const
+{
+  if(mPlugin)
+  {
+    return mPlugin->GetNumberOfPointsPerOneUnitOfPointSize();
+  }
+  return TextAbstraction::FontClient::NUMBER_OF_POINTS_PER_ONE_UNIT_OF_POINT_SIZE;
+  ;
+}
+
+FT_FaceRec_* FontClient::GetFreetypeFace(FontId fontId)
+{
+  CreatePlugin();
+
+  return mPlugin->GetFreetypeFace(fontId);
 }
 
-FT_FaceRec_* FontClient::GetFreetypeFace( FontId fontId )
+FontDescription::Type FontClient::GetFontType(FontId fontId)
 {
   CreatePlugin();
 
-  return mPlugin->GetFreetypeFace( fontId );
+  return mPlugin->GetFontType(fontId);
 }
 
-FontDescription::Type FontClient::GetFontType( FontId fontId )
+bool FontClient::AddCustomFontDirectory(const FontPath& path)
 {
   CreatePlugin();
 
-  return mPlugin->GetFontType( fontId );
+  return mPlugin->AddCustomFontDirectory(path);
 }
 
-bool FontClient::AddCustomFontDirectory( const FontPath& path )
+HarfBuzzFontHandle FontClient::GetHarfBuzzFont(FontId fontId)
 {
   CreatePlugin();
 
-  return mPlugin->AddCustomFontDirectory( path );
+  return mPlugin->GetHarfBuzzFont(fontId);
 }
 
 void FontClient::CreatePlugin()
 {
-  if( !mPlugin )
+  std::scoped_lock lock(gMutex);
+  if(!mPlugin)
   {
-    mPlugin = new Plugin( mDpiHorizontal, mDpiVertical );
+    mPlugin = new Plugin(mDpiHorizontal, mDpiVertical);
   }
 }