From 70608ec03d747812f29245a4e74715dbc9e92ed0 Mon Sep 17 00:00:00 2001 From: kgv Date: Thu, 21 Nov 2019 12:32:56 +0300 Subject: [PATCH] 0030706: Visualization - fetch font folder list from fontconfig library on Linux [backport occt720] vfont command now prints fonts in alphabetical order. # Conflicts: # adm/qmake/OccToolkit.pri --- adm/cmake/occt_csf.cmake | 3 +- adm/genproj.tcl | 1 + src/Font/Font_FontMgr.cxx | 129 +++++++++++-------- src/TKService/EXTERNLIB | 1 + src/ViewerTest/ViewerTest_ObjectCommands.cxx | 19 ++- 5 files changed, 99 insertions(+), 54 deletions(-) diff --git a/adm/cmake/occt_csf.cmake b/adm/cmake/occt_csf.cmake index c62425da80..07fe2ead23 100644 --- a/adm/cmake/occt_csf.cmake +++ b/adm/cmake/occt_csf.cmake @@ -79,7 +79,7 @@ if (WIN32) set (CSF_OpenGlLibs "opengl32.lib") endif() - else() +else() if (APPLE) set (CSF_objc "objc") @@ -112,5 +112,6 @@ if (WIN32) set (CSF_OpenGlLibs "GL") set (CSF_XwLibs "X11 Xext Xmu Xi") set (CSF_dl "dl") + set (CSF_fontconfig "fontconfig") endif() endif() diff --git a/adm/genproj.tcl b/adm/genproj.tcl index 2bc4f38844..d0cc0b1f8a 100644 --- a/adm/genproj.tcl +++ b/adm/genproj.tcl @@ -1375,6 +1375,7 @@ proc osutils:csfList { theOS theCsfLibsMap theCsfFrmsMap } { set aLibsMap(CSF_TclTkLibs) "" set aLibsMap(CSF_QT) "QtCore QtGui" } else { + set aLibsMap(CSF_fontconfig) "fontconfig" if { "$theOS" == "qnx" } { # CSF_ThreadLibs - pthread API is part of libc on QNX set aLibsMap(CSF_OpenGlLibs) "EGL GLESv2" diff --git a/src/Font/Font_FontMgr.cxx b/src/Font/Font_FontMgr.cxx index 3fef8bdb77..4fad2f6d76 100644 --- a/src/Font/Font_FontMgr.cxx +++ b/src/Font/Font_FontMgr.cxx @@ -88,6 +88,10 @@ IMPLEMENT_STANDARD_RTTIEXT(Font_FontMgr,Standard_Transient) "/usr/X11/lib/X11/fs/config", NULL }; + + // Although fontconfig library can be built for various platforms, + // practically it is useful only on desktop Linux distributions, where it is always packaged. + #include #endif #ifdef __APPLE__ @@ -393,7 +397,7 @@ Standard_Boolean Font_FontMgr::RegisterFont (const Handle(Font_SystemFont)& theF void Font_FontMgr::InitFontDataBase() { myFontMap.Clear(); - Handle(Font_FTLibrary) aFtLibrary; + Handle(Font_FTLibrary) aFtLibrary = new Font_FTLibrary(); #if defined(OCCT_UWP) // system font files are not accessible @@ -428,7 +432,6 @@ void Font_FontMgr::InitFontDataBase() aSupportedExtensions.Add (TCollection_AsciiString (anExt)); } - aFtLibrary = new Font_FTLibrary(); static const DWORD aBufferSize = 256; char aNameBuff[aBufferSize]; char aPathBuff[aBufferSize]; @@ -471,61 +474,86 @@ void Font_FontMgr::InitFontDataBase() NCollection_Map aMapOfFontsDirs; #if !defined(__ANDROID__) && !defined(__APPLE__) - const OSD_Protection aProtectRead (OSD_R, OSD_R, OSD_R, OSD_R); - - // read fonts directories from font service config file (obsolete) - for (Standard_Integer anIter = 0; myFontServiceConf[anIter] != NULL; ++anIter) + if (FcConfig* aFcCfg = FcInitLoadConfig()) { - const TCollection_AsciiString aFileOfFontsPath (myFontServiceConf[anIter]); - OSD_File aFile (aFileOfFontsPath); - if (!aFile.Exists()) + if (FcStrList* aFcFontDir = FcConfigGetFontDirs (aFcCfg)) { - continue; - } - - aFile.Open (OSD_ReadOnly, aProtectRead); - if (!aFile.IsOpen()) - { - continue; - } - - Standard_Integer aNByte = 256; - Standard_Integer aNbyteRead; - TCollection_AsciiString aStr; // read string with information - while (!aFile.IsAtEnd()) - { - Standard_Integer aLocation = -1; - Standard_Integer aPathLocation = -1; - - aFile.ReadLine (aStr, aNByte, aNbyteRead); // reading 1 line (256 bytes) - aLocation = aStr.Search ("catalogue="); - if (aLocation < 0) + for (;;) { - aLocation = aStr.Search ("catalogue ="); - } - - aPathLocation = aStr.Search ("/"); - if (aLocation > 0 && aPathLocation > 0) - { - aStr = aStr.Split (aPathLocation - 1); - TCollection_AsciiString aFontPath; - Standard_Integer aPathNumber = 1; - do + FcChar8* aFcFolder = FcStrListNext (aFcFontDir); + if (aFcFolder == NULL) { - // Getting directory paths, which can be splitted by "," or ":" - aFontPath = aStr.Token (":,", aPathNumber); - aFontPath.RightAdjust(); - if (!aFontPath.IsEmpty()) - { - OSD_Path aPath(aFontPath); - addDirsRecursively (aPath, aMapOfFontsDirs); - } - aPathNumber++; + break; } - while (!aFontPath.IsEmpty()); + + TCollection_AsciiString aPathStr ((const char* )aFcFolder); + OSD_Path aPath (aPathStr); + addDirsRecursively (aPath, aMapOfFontsDirs); } + FcStrListDone (aFcFontDir); + } + FcConfigDestroy (aFcCfg); + } + + const OSD_Protection aProtectRead (OSD_R, OSD_R, OSD_R, OSD_R); + if (aMapOfFontsDirs.IsEmpty()) + { + Message::DefaultMessenger()->Send ("Font_FontMgr, fontconfig library returns an empty folder list", Message_Alarm); + + // read fonts directories from font service config file (obsolete) + for (Standard_Integer anIter = 0; myFontServiceConf[anIter] != NULL; ++anIter) + { + const TCollection_AsciiString aFileOfFontsPath (myFontServiceConf[anIter]); + OSD_File aFile (aFileOfFontsPath); + if (!aFile.Exists()) + { + continue; + } + + aFile.Open (OSD_ReadOnly, aProtectRead); + if (!aFile.IsOpen()) + { + continue; + } + + Standard_Integer aNByte = 256; + Standard_Integer aNbyteRead; + TCollection_AsciiString aStr; // read string with information + while (!aFile.IsAtEnd()) + { + Standard_Integer aLocation = -1; + Standard_Integer aPathLocation = -1; + + aFile.ReadLine (aStr, aNByte, aNbyteRead); // reading 1 line (256 bytes) + aLocation = aStr.Search ("catalogue="); + if (aLocation < 0) + { + aLocation = aStr.Search ("catalogue ="); + } + + aPathLocation = aStr.Search ("/"); + if (aLocation > 0 && aPathLocation > 0) + { + aStr = aStr.Split (aPathLocation - 1); + TCollection_AsciiString aFontPath; + Standard_Integer aPathNumber = 1; + do + { + // Getting directory paths, which can be splitted by "," or ":" + aFontPath = aStr.Token (":,", aPathNumber); + aFontPath.RightAdjust(); + if (!aFontPath.IsEmpty()) + { + OSD_Path aPath(aFontPath); + addDirsRecursively (aPath, aMapOfFontsDirs); + } + aPathNumber++; + } + while (!aFontPath.IsEmpty()); + } + } + aFile.Close(); } - aFile.Close(); } #endif @@ -545,7 +573,6 @@ void Font_FontMgr::InitFontDataBase() aSupportedExtensions.Add (TCollection_AsciiString (anExt)); } - aFtLibrary = new Font_FTLibrary(); for (NCollection_Map::Iterator anIter (aMapOfFontsDirs); anIter.More(); anIter.Next()) { diff --git a/src/TKService/EXTERNLIB b/src/TKService/EXTERNLIB index 32aba57c23..5bf60ae72f 100755 --- a/src/TKService/EXTERNLIB +++ b/src/TKService/EXTERNLIB @@ -10,6 +10,7 @@ CSF_OpenGlLibs CSF_XwLibs CSF_dpsLibs CSF_XmuLibs +CSF_fontconfig CSF_objc CSF_Appkit CSF_IOKit diff --git a/src/ViewerTest/ViewerTest_ObjectCommands.cxx b/src/ViewerTest/ViewerTest_ObjectCommands.cxx index 78b69eefe0..4161bab034 100644 --- a/src/ViewerTest/ViewerTest_ObjectCommands.cxx +++ b/src/ViewerTest/ViewerTest_ObjectCommands.cxx @@ -5526,6 +5526,14 @@ static int TextToBRep (Draw_Interpretor& /*theDI*/, //function : VFont //purpose : Font management //======================================================================= +struct FontComparator +{ + bool operator() (const Handle(Font_SystemFont)& theFontA, + const Handle(Font_SystemFont)& theFontB) + { + return theFontA->FontKey().IsLess (theFontB->FontKey()); + } +}; static int VFont (Draw_Interpretor& theDI, Standard_Integer theArgNb, @@ -5537,9 +5545,16 @@ static int VFont (Draw_Interpretor& theDI, // just print the list of available fonts Standard_Boolean isFirst = Standard_True; const Font_NListOfSystemFont aFonts = aMgr->GetAvailableFonts(); - for (Font_NListOfSystemFont::Iterator anIter (aFonts); anIter.More(); anIter.Next()) + std::vector aFontsSorted; + aFontsSorted.reserve (aFonts.Size()); + for (Font_NListOfSystemFont::Iterator aFontIter (aFonts); aFontIter.More(); aFontIter.Next()) { - const Handle(Font_SystemFont)& aFont = anIter.Value(); + aFontsSorted.push_back (aFontIter.Value()); + } + std::stable_sort (aFontsSorted.begin(), aFontsSorted.end(), FontComparator()); + for (std::vector::iterator aFontIter = aFontsSorted.begin(); aFontIter != aFontsSorted.end(); ++aFontIter) + { + const Handle(Font_SystemFont)& aFont = *aFontIter; if (!isFirst) { theDI << "\n";