diff --git a/data/wqy-microhei.ttf b/data/wqy-microhei.ttf new file mode 100644 index 0000000000..2c9bc2d4e0 Binary files /dev/null and b/data/wqy-microhei.ttf differ diff --git a/iphone/Maps/Classes/EAGLView.mm b/iphone/Maps/Classes/EAGLView.mm index 6582a10f2b..a901bbac8b 100644 --- a/iphone/Maps/Classes/EAGLView.mm +++ b/iphone/Maps/Classes/EAGLView.mm @@ -80,9 +80,11 @@ resourceManager = shared_ptr(new yg::ResourceManager( 15000 * sizeof(yg::gl::Vertex), 30000 * sizeof(unsigned short), 20, 1500 * sizeof(yg::gl::Vertex), 3000 * sizeof(unsigned short), 100, - 512, 256, 10)); + 512, 256, 10, + 2000000)); - resourceManager->addFont(GetPlatform().ReadPathForFile("dejavusans.ttf").c_str()); +// resourceManager->addFont(GetPlatform().ReadPathForFile("dejavusans.ttf").c_str()); + resourceManager->addFont(GetPlatform().ReadPathForFile("wqy-microhei.ttf").c_str()); drawer = shared_ptr(new DrawerYG(resourceManager, GetPlatform().SkinName(), !GetPlatform().IsMultiSampled())); drawer->setFrameBuffer(frameBuffer); diff --git a/iphone/Maps/Maps.xcodeproj/project.pbxproj b/iphone/Maps/Maps.xcodeproj/project.pbxproj index 1a29c3f06f..0e481c56a0 100644 --- a/iphone/Maps/Maps.xcodeproj/project.pbxproj +++ b/iphone/Maps/Maps.xcodeproj/project.pbxproj @@ -26,6 +26,7 @@ EE12020F11CD464100ABDD5D /* libindexer.a in Frameworks */ = {isa = PBXBuildFile; fileRef = EE12020511CD464100ABDD5D /* libindexer.a */; }; EE12021011CD464100ABDD5D /* libmap.a in Frameworks */ = {isa = PBXBuildFile; fileRef = EE12020611CD464100ABDD5D /* libmap.a */; }; EE12021211CD464100ABDD5D /* libyg.a in Frameworks */ = {isa = PBXBuildFile; fileRef = EE12020811CD464100ABDD5D /* libyg.a */; }; + EE12092C12BD67C900068DC3 /* wqy-microhei.ttf in Resources */ = {isa = PBXBuildFile; fileRef = EE12092B12BD67C900068DC3 /* wqy-microhei.ttf */; }; EE7F29811219ECA300EB67A9 /* RenderBuffer.mm in Sources */ = {isa = PBXBuildFile; fileRef = EE7F297D1219ECA300EB67A9 /* RenderBuffer.mm */; }; EE7F29821219ECA300EB67A9 /* RenderContext.mm in Sources */ = {isa = PBXBuildFile; fileRef = EE7F297E1219ECA300EB67A9 /* RenderContext.mm */; }; EE7F29831219ECA300EB67A9 /* WindowHandle.mm in Sources */ = {isa = PBXBuildFile; fileRef = EE7F29801219ECA300EB67A9 /* WindowHandle.mm */; }; @@ -85,6 +86,7 @@ EE12020511CD464100ABDD5D /* libindexer.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; path = libindexer.a; sourceTree = SOURCE_ROOT; }; EE12020611CD464100ABDD5D /* libmap.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; path = libmap.a; sourceTree = SOURCE_ROOT; }; EE12020811CD464100ABDD5D /* libyg.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; path = libyg.a; sourceTree = SOURCE_ROOT; }; + EE12092B12BD67C900068DC3 /* wqy-microhei.ttf */ = {isa = PBXFileReference; lastKnownFileType = file; name = "wqy-microhei.ttf"; path = "../../data/wqy-microhei.ttf"; sourceTree = ""; }; EE16192B126E374500622BD0 /* RenderContext.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = RenderContext.hpp; path = Classes/RenderContext.hpp; sourceTree = SOURCE_ROOT; }; EE7F297C1219ECA300EB67A9 /* RenderBuffer.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; name = RenderBuffer.hpp; path = Classes/RenderBuffer.hpp; sourceTree = SOURCE_ROOT; }; EE7F297D1219ECA300EB67A9 /* RenderBuffer.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; name = RenderBuffer.mm; path = Classes/RenderBuffer.mm; sourceTree = SOURCE_ROOT; }; @@ -238,6 +240,7 @@ FA065FC61286143F00FEA989 /* External Resources */ = { isa = PBXGroup; children = ( + EE12092B12BD67C900068DC3 /* wqy-microhei.ttf */, EEFC0A9512B561B7002914FF /* dejavusans.ttf */, EEE4C9411298A31B007231A9 /* basic_highres.skn */, EEE4C93F1298A303007231A9 /* symbols_48.png */, @@ -371,6 +374,7 @@ EEE4C9401298A303007231A9 /* symbols_48.png in Resources */, EEE4C9421298A31B007231A9 /* basic_highres.skn in Resources */, EEFC0A9612B561B7002914FF /* dejavusans.ttf in Resources */, + EE12092C12BD67C900068DC3 /* wqy-microhei.ttf in Resources */, ); runOnlyForDeploymentPostprocessing = 0; }; diff --git a/qt/qt.pro b/qt/qt.pro index 71101417a3..cb6a78ac68 100644 --- a/qt/qt.pro +++ b/qt/qt.pro @@ -29,8 +29,7 @@ macx { # Bundle Resouces CLASSIFICATOR_RESOURCES.files = ../data/classificator.txt ../data/drawing_rules.bin ../data/visibility.txt CLASSIFICATOR_RESOURCES.path = Contents/Resources - SKIN_RESOURCES.files = ../data/basic.skn ../data/symbols_24.png \ - ../data/dejavusans.ttf + SKIN_RESOURCES.files = ../data/basic.skn ../data/symbols_24.png ../data/wqy-microhei.ttf SKIN_RESOURCES.path = Contents/Resources QMAKE_BUNDLE_DATA += CLASSIFICATOR_RESOURCES SKIN_RESOURCES } diff --git a/qt/widgets.cpp b/qt/widgets.cpp index 23c06d4957..0866fbf68b 100644 --- a/qt/widgets.cpp +++ b/qt/widgets.cpp @@ -41,9 +41,11 @@ namespace qt 10000 * sizeof(unsigned short), 100, 512, 256, - 15)); + 15, + 2000000)); - m_resourceManager->addFont(GetPlatform().ReadPathForFile("dejavusans.ttf").c_str()); +// m_resourceManager->addFont(GetPlatform().ReadPathForFile("dejavusans.ttf").c_str()); + m_resourceManager->addFont(GetPlatform().ReadPathForFile("wqy-microhei.ttf").c_str()); m_p = shared_ptr(new DrawerYG(m_resourceManager, GetPlatform().SkinName(), !GetPlatform().IsMultiSampled())); m_p->setFrameBuffer(make_shared_ptr(new yg::gl::FrameBuffer(true))); diff --git a/qt_tstfrm/tstwidgets.cpp b/qt_tstfrm/tstwidgets.cpp index 7525981e63..c7ef09c946 100644 --- a/qt_tstfrm/tstwidgets.cpp +++ b/qt_tstfrm/tstwidgets.cpp @@ -46,9 +46,11 @@ void GLDrawWidget::initializeGL() 3000 * sizeof(yg::gl::Vertex), 5000 * sizeof(unsigned short), 100, - 512, 256, 15)); + 512, 256, 15, + 2000000)); - m_resourceManager->addFont(GetPlatform().ReadPathForFile("dejavusans.ttf").c_str()); +// m_resourceManager->addFont(GetPlatform().ReadPathForFile("dejavusans.ttf").c_str()); + m_resourceManager->addFont(GetPlatform().ReadPathForFile("wqy-microhei.ttf").c_str()); /* m_resourceManager = make_shared_ptr(new yg::ResourceManager( 5000 * sizeof(yg::gl::Vertex), 10000 * sizeof(unsigned short), diff --git a/yg/ft2_debug.cpp b/yg/ft2_debug.cpp new file mode 100644 index 0000000000..f9391aa864 --- /dev/null +++ b/yg/ft2_debug.cpp @@ -0,0 +1,35 @@ +#include "../base/SRC_FIRST.hpp" +#include "ft2_debug.hpp" + +#undef __FTERRORS_H__ +#define FT_ERRORDEF( e, v, s ) { e, s }, +#define FT_ERROR_START_LIST { +#define FT_ERROR_END_LIST { 0, 0 } }; + +const struct +{ + int err_code; + const char* err_msg; +} ft_errors[] = + +#include FT_ERRORS_H + +char const * FT_Error_Description(FT_Error error) +{ + int i = 1; + while (ft_errors[i].err_code != 0) + { + if (ft_errors[i].err_code == error) + break; + else + ++i; + } + return ft_errors[i].err_msg; +} + +void CheckError(FT_Error error) +{ + if (error != 0) + LOG(LINFO, ("FT_Error : ", FT_Error_Description(error))); +} + diff --git a/yg/ft2_debug.hpp b/yg/ft2_debug.hpp new file mode 100644 index 0000000000..f296ac875f --- /dev/null +++ b/yg/ft2_debug.hpp @@ -0,0 +1,16 @@ +#pragma once + +#include +#include +#include + +#include "../base/logging.hpp" + +#include FT_FREETYPE_H +#include FT_STROKER_H +#include FT_GLYPH_H + +char const * FT_Error_Description(FT_Error error); +void CheckError(FT_Error error); + +#define FTCHECK(x) do {FT_Error e = (x); CheckError(e);} while (false) diff --git a/yg/glyph_cache.cpp b/yg/glyph_cache.cpp index 67c711d213..634a012f5a 100644 --- a/yg/glyph_cache.cpp +++ b/yg/glyph_cache.cpp @@ -1,57 +1,21 @@ #include "../base/SRC_FIRST.hpp" #include "../base/logging.hpp" +#include "../base/ptr_utils.hpp" #include "../coding/lodepng_io.hpp" #include "../std/vector.hpp" #include "../std/map.hpp" #include #include "glyph_cache.hpp" +#include "glyph_cache_impl.hpp" #include "data_formats.hpp" #include "internal/opengl.hpp" - -#include - -#include FT_FREETYPE_H -#include FT_STROKER_H -#include FT_GLYPH_H - -#undef __FTERRORS_H__ -#define FT_ERRORDEF( e, v, s ) { e, s }, -#define FT_ERROR_START_LIST { -#define FT_ERROR_END_LIST { 0, 0 } }; - -const struct -{ - int err_code; - const char* err_msg; -} ft_errors[] = - -#include FT_ERRORS_H - -void CheckError(FT_Error error) -{ - if (error != 0) - { - int i = 0; - while (ft_errors[i].err_code != 0) - { - if (ft_errors[i].err_code == error) - { - LOG(LINFO, ("FT_Error : ", ft_errors[i].err_msg)); - break; - } - ++i; - } - } -} - -#define FTCHECK(x) do {FT_Error e = (x); CheckError(e);} while (false) +#include "ft2_debug.hpp" namespace gil = boost::gil; namespace yg { - - GlyphKey::GlyphKey(unsigned short id, int fontSize, bool isMask) + GlyphKey::GlyphKey(int id, int fontSize, bool isMask) : m_id(id), m_fontSize(fontSize), m_isMask(isMask) {} @@ -64,136 +28,120 @@ namespace yg return l.m_isMask < r.m_isMask; } - struct GlyphCache::Impl + GlyphCache::GlyphCache(size_t maxSize) : m_impl(new GlyphCacheImpl(maxSize)) { - FT_Library m_lib; - FT_Stroker m_stroker; - - typedef vector TFontFaces; - TFontFaces m_faces; - - typedef map > TGlyphs; - TGlyphs m_glyphs; - }; - - GlyphCache::GlyphCache() : m_impl(new Impl()) - { - FTCHECK(FT_Init_FreeType(&m_impl->m_lib)); - FTCHECK(FT_Stroker_New(m_impl->m_lib, &m_impl->m_stroker)); - FT_Stroker_Set(m_impl->m_stroker, 2 * 64, FT_STROKER_LINECAP_ROUND, FT_STROKER_LINEJOIN_ROUND, 0); - } - - GlyphCache::~GlyphCache() - { - for (Impl::TFontFaces::iterator it = m_impl->m_faces.begin(); it != m_impl->m_faces.end(); ++it) - FT_Done_Face(*it); - FT_Stroker_Done(m_impl->m_stroker); - FT_Done_FreeType(m_impl->m_lib); } void GlyphCache::addFont(char const * fileName) { - FT_Face face; - FT_Error err = FT_New_Face(m_impl->m_lib, fileName, 0, &face); - m_impl->m_faces.push_back(face); - if (err == 0) - LOG(LINFO, ("Added font ", fileName)) - else - LOG(LINFO, ("Font wasn't added! Path: ", fileName, ", Error: ", err)); + m_impl->m_fonts.push_back(make_shared_ptr(new Font(fileName))); } - shared_ptr const GlyphCache::getGlyph(GlyphKey const & k) + shared_ptr const GlyphCache::getGlyph(GlyphKey const & key) { - GlyphKey key(k); - Impl::TGlyphs::const_iterator it = m_impl->m_glyphs.find(key); - if (it != m_impl->m_glyphs.end()) - return it->second; - else + Font * font = m_impl->m_fonts.back().get(); + FTC_FaceID faceID = reinterpret_cast(font); + FTC_ScalerRec fontScaler = { - FT_Face face = m_impl->m_faces.front(); - FTCHECK(FT_Set_Pixel_Sizes(face, 0, key.m_fontSize)); + faceID, + key.m_fontSize, + key.m_fontSize, + 1, + 0, + 0 + }; - int symbolIdx = FT_Get_Char_Index(face, key.m_id); - if (symbolIdx == 0) - { - key = GlyphKey(65533, key.m_fontSize, key.m_isMask); - it = m_impl->m_glyphs.find(key); - if (it != m_impl->m_glyphs.end()) - return it->second; - else - { - symbolIdx = FT_Get_Char_Index(face, key.m_id); - if (symbolIdx == 0) - throw std::exception(); - } - } + int charIDX = FTC_CMapCache_Lookup( + m_impl->m_charMapCache, + faceID, + -1, + key.m_id + ); - FTCHECK(FT_Load_Glyph(face, symbolIdx, FT_LOAD_DEFAULT)); + if (charIDX == 0) + { + charIDX = FTC_CMapCache_Lookup( + m_impl->m_charMapCache, + faceID, + -1, + 65533 + ); + if (charIDX == 0) + charIDX = FTC_CMapCache_Lookup( + m_impl->m_charMapCache, + faceID, + -1, + 32 + ); + } - FT_Glyph glyph; - FT_BitmapGlyph bitmapGlyph; + FT_Glyph glyph = 0; + FTC_Node glyphNode; - if (key.m_isMask) - { - FTCHECK(FT_Get_Glyph(face->glyph, &glyph)); - FTCHECK(FT_Glyph_Stroke(&glyph, m_impl->m_stroker, 1)); - FTCHECK(FT_Glyph_To_Bitmap(&glyph, FT_RENDER_MODE_NORMAL, 0, 0)); - bitmapGlyph = (FT_BitmapGlyph)glyph; - } - else - { - FTCHECK(FT_Render_Glyph(face->glyph, FT_RENDER_MODE_NORMAL)); - FTCHECK(FT_Get_Glyph(face->glyph, &glyph)); - bitmapGlyph = (FT_BitmapGlyph)glyph; - } + if (key.m_isMask) + FTCHECK(FTC_ImageCache_LookupScaler( + m_impl->m_strokedGlyphCache, + &fontScaler, + FT_LOAD_DEFAULT, + charIDX, + &glyph, + 0 + )); + else + FTCHECK(FTC_ImageCache_LookupScaler( + m_impl->m_normalGlyphCache, + &fontScaler, + FT_LOAD_DEFAULT | FT_LOAD_RENDER, + charIDX, + &glyph, + 0 + )); - shared_ptr info(new GlyphInfo()); - info->m_height = bitmapGlyph->bitmap.rows; - info->m_width = bitmapGlyph->bitmap.width; - info->m_xOffset = bitmapGlyph->left; - info->m_yOffset = bitmapGlyph->top - info->m_height; - info->m_xAdvance = int(bitmapGlyph->root.advance.x >> 16); - info->m_color = key.m_isMask ? yg::Color(255, 255, 255, 0) : yg::Color(0, 0, 0, 0); + FT_BitmapGlyph bitmapGlyph = (FT_BitmapGlyph)glyph; - if ((info->m_width != 0) && (info->m_height != 0)) - { - info->m_bitmap.resize(info->m_width * info->m_height * sizeof(DATA_TRAITS::pixel_t)); + shared_ptr info(new GlyphInfo()); - DATA_TRAITS::view_t dstView = gil::interleaved_view( + info->m_height = bitmapGlyph ? bitmapGlyph->bitmap.rows : 0; + info->m_width = bitmapGlyph ? bitmapGlyph->bitmap.width : 0; + info->m_xOffset = bitmapGlyph ? bitmapGlyph->left : 0; + info->m_yOffset = bitmapGlyph ? bitmapGlyph->top - info->m_height : 0; + info->m_xAdvance = bitmapGlyph ? int(bitmapGlyph->root.advance.x >> 16) : 0; + info->m_color = key.m_isMask ? yg::Color(255, 255, 255, 0) : yg::Color(0, 0, 0, 0); + + if ((info->m_width != 0) && (info->m_height != 0)) + { + info->m_bitmap.resize(info->m_width * info->m_height * sizeof(DATA_TRAITS::pixel_t)); + + DATA_TRAITS::view_t dstView = gil::interleaved_view( info->m_width, info->m_height, (DATA_TRAITS::pixel_t*)&info->m_bitmap[0], info->m_width * sizeof(DATA_TRAITS::pixel_t) ); - gil::gray8c_view_t srcView = gil::interleaved_view( - info->m_width, - info->m_height, - (gil::gray8_pixel_t*)bitmapGlyph->bitmap.buffer, - bitmapGlyph->bitmap.pitch - ); + gil::gray8c_view_t srcView = gil::interleaved_view( + info->m_width, + info->m_height, + (gil::gray8_pixel_t*)bitmapGlyph->bitmap.buffer, + bitmapGlyph->bitmap.pitch + ); - DATA_TRAITS::pixel_t c; + DATA_TRAITS::pixel_t c; - gil::get_color(c, gil::red_t()) = key.m_isMask ? DATA_TRAITS::maxChannelVal : 0; - gil::get_color(c, gil::green_t()) = key.m_isMask ? DATA_TRAITS::maxChannelVal : 0; - gil::get_color(c, gil::blue_t()) = key.m_isMask ? DATA_TRAITS::maxChannelVal : 0; - gil::get_color(c, gil::alpha_t()) = 0; + gil::get_color(c, gil::red_t()) = key.m_isMask ? DATA_TRAITS::maxChannelVal : 0; + gil::get_color(c, gil::green_t()) = key.m_isMask ? DATA_TRAITS::maxChannelVal : 0; + gil::get_color(c, gil::blue_t()) = key.m_isMask ? DATA_TRAITS::maxChannelVal : 0; + gil::get_color(c, gil::alpha_t()) = 0; - for (size_t y = 0; y < srcView.height(); ++y) - for (size_t x = 0; x < srcView.width(); ++x) - { - gil::get_color(c, gil::alpha_t()) = srcView(x, y) / DATA_TRAITS::channelScaleFactor; - dstView(x, y) = c; - } - } - - FT_Done_Glyph(glyph); - - m_impl->m_glyphs[key] = info; + for (size_t y = 0; y < srcView.height(); ++y) + for (size_t x = 0; x < srcView.width(); ++x) + { + gil::get_color(c, gil::alpha_t()) = srcView(x, y) / DATA_TRAITS::channelScaleFactor; + dstView(x, y) = c; + } } - return m_impl->m_glyphs[key]; + return info; } void GlyphInfo::dump(const char *fileName) diff --git a/yg/glyph_cache.hpp b/yg/glyph_cache.hpp index 05100b768b..e2b1951bab 100644 --- a/yg/glyph_cache.hpp +++ b/yg/glyph_cache.hpp @@ -21,26 +21,27 @@ namespace yg struct GlyphKey { - unsigned short m_id; + int m_id; int m_fontSize; bool m_isMask; - GlyphKey(unsigned short id, int fontSize, bool isMask); + GlyphKey(int id, int fontSize, bool isMask); }; bool operator<(GlyphKey const & l, GlyphKey const & r); + class GlyphCacheImpl; + class GlyphCache { private: - struct Impl; - shared_ptr m_impl; + shared_ptr m_impl; public: - GlyphCache(); - ~GlyphCache(); + GlyphCache(size_t maxSize); + void reset(); void addFont(char const * fileName); shared_ptr const getGlyph(GlyphKey const & key); diff --git a/yg/glyph_cache_impl.cpp b/yg/glyph_cache_impl.cpp new file mode 100644 index 0000000000..20c9e7519a --- /dev/null +++ b/yg/glyph_cache_impl.cpp @@ -0,0 +1,62 @@ +#include "../base/SRC_FIRST.hpp" +#include "glyph_cache_impl.hpp" + +#include <../cache/ftcglyph.h> +#include <../cache/ftcimage.h> +#include <../cache/ftcsbits.h> +#include <../cache/ftccback.h> +#include <../cache/ftccache.h> + +namespace yg +{ + Font::Font(char const * name) : m_name(name) + { + FILE * fp = fopen(name, "rb"); + fseek(fp, 0, SEEK_END); + int size = ftell(fp); + fseek(fp, 0, SEEK_SET); + + m_fontData.resize(size); + fread(&m_fontData[0], 1, size, fp); + + fclose(fp); + } + + FT_Error Font::CreateFaceID(FT_Library library, FT_Face *face) + { + return FT_New_Memory_Face(library, &m_fontData[0], m_fontData.size(), 0, face); + } + + GlyphCacheImpl::GlyphCacheImpl(size_t maxSize) + { + FTCHECK(FT_Init_FreeType(&m_lib)); + + /// Initializing caches + FTCHECK(FTC_Manager_New(m_lib, 3, 10, maxSize, &RequestFace, 0, &m_manager)); + + FTCHECK(FTC_ImageCache_New(m_manager, &m_normalGlyphCache)); + + /// Initializing stroker + FTCHECK(FT_Stroker_New(m_lib, &m_stroker)); + FT_Stroker_Set(m_stroker, 2 * 64, FT_STROKER_LINECAP_ROUND, FT_STROKER_LINEJOIN_ROUND, 0); + + FTCHECK(FTC_StrokedImageCache_New(m_manager, &m_strokedGlyphCache, m_stroker)); + + FTCHECK(FTC_CMapCache_New(m_manager, &m_charMapCache)); + + } + + GlyphCacheImpl::~GlyphCacheImpl() + { + FTC_Manager_Done(m_manager); + FT_Stroker_Done(m_stroker); + FT_Done_FreeType(m_lib); + } + + FT_Error GlyphCacheImpl::RequestFace(FTC_FaceID faceID, FT_Library library, FT_Pointer requestData, FT_Face * face) + { + //GlyphCacheImpl * glyphCacheImpl = reinterpret_cast(requestData); + Font * font = reinterpret_cast(faceID); + return font->CreateFaceID(library, face); + } +} diff --git a/yg/glyph_cache_impl.hpp b/yg/glyph_cache_impl.hpp new file mode 100644 index 0000000000..3f056b2651 --- /dev/null +++ b/yg/glyph_cache_impl.hpp @@ -0,0 +1,43 @@ +#pragma once + +#include "ft2_debug.hpp" +#include "../std/string.hpp" +#include "../std/vector.hpp" + + + +namespace yg +{ + struct Font + { + string m_name; + vector m_fontData; + /// information about symbol ranges + /// ... + /// constructor + Font(char const * name); + + FT_Error CreateFaceID(FT_Library library, FT_Face * face); + }; + + struct GlyphCacheImpl + { + FT_Library m_lib; + FT_Stroker m_stroker; + + FTC_Manager m_manager; + + FTC_ImageCache m_strokedGlyphCache; + FTC_ImageCache m_normalGlyphCache; + + FTC_CMapCache m_charMapCache; + + typedef vector TFonts; + TFonts m_fonts; + + static FT_Error RequestFace(FTC_FaceID faceID, FT_Library library, FT_Pointer requestData, FT_Face * face); + + GlyphCacheImpl(size_t maxSize); + ~GlyphCacheImpl(); + }; +} diff --git a/yg/resource_manager.cpp b/yg/resource_manager.cpp index 1143675e69..25a954fa67 100644 --- a/yg/resource_manager.cpp +++ b/yg/resource_manager.cpp @@ -18,7 +18,8 @@ namespace yg ResourceManager::ResourceManager(size_t vbSize, size_t ibSize, size_t storagesCount, size_t smallVBSize, size_t smallIBSize, size_t smallStoragesCount, - size_t texWidth, size_t texHeight, size_t texCount) + size_t texWidth, size_t texHeight, size_t texCount, + size_t maxGlyphCacheSize) : m_glyphCache(maxGlyphCacheSize) { for (size_t i = 0; i < storagesCount; ++i) m_storages.push_back(gl::Storage(vbSize, ibSize)); diff --git a/yg/resource_manager.hpp b/yg/resource_manager.hpp index dbe037ea67..b9f61f6c8d 100644 --- a/yg/resource_manager.hpp +++ b/yg/resource_manager.hpp @@ -46,7 +46,8 @@ namespace yg ResourceManager(size_t vbSize, size_t ibSize, size_t storagesCount, size_t smallVBSize, size_t smallIBSize, size_t smallStoragesCount, - size_t texWidth, size_t texHeight, size_t texCount); + size_t texWidth, size_t texHeight, size_t texCount, + size_t maxGlyphCacheSize); shared_ptr const & getTexture(string const & fileName); diff --git a/yg/skin_page.cpp b/yg/skin_page.cpp index 47a438cf11..38001910c6 100644 --- a/yg/skin_page.cpp +++ b/yg/skin_page.cpp @@ -392,9 +392,9 @@ namespace yg for (size_t y = 2; y < rect.SizeY() - 2; ++y) for (size_t x = 2; x < rect.SizeX() - 2; ++x) v(x, y) = srcView(x - 2, y - 2); - - dynTexture->upload(&v(0, 0), rect); } + + dynTexture->upload(&v(0, 0), rect); } m_glyphUploadCommands.clear(); diff --git a/yg/yg.pro b/yg/yg.pro index 05b48935f1..4863ca473a 100644 --- a/yg/yg.pro +++ b/yg/yg.pro @@ -54,7 +54,9 @@ SOURCES += \ skin_page.cpp \ storage.cpp \ render_state_updater.cpp \ - glyph_cache.cpp + glyph_cache.cpp \ + glyph_cache_impl.cpp \ + ft2_debug.cpp HEADERS += \ internal/opengl.hpp \ @@ -91,7 +93,9 @@ HEADERS += \ render_state_updater.hpp \ render_target.hpp \ glyph_cache.hpp \ - data_formats.hpp + data_formats.hpp \ + glyph_cache_impl.hpp \ + ft2_debug.hpp !iphonesimulator-g++42 { !iphonedevice-g++42 { diff --git a/yg/yg_tests/glyph_cache_test.cpp b/yg/yg_tests/glyph_cache_test.cpp index 9e79084f6b..5872a87ae9 100644 --- a/yg/yg_tests/glyph_cache_test.cpp +++ b/yg/yg_tests/glyph_cache_test.cpp @@ -5,10 +5,10 @@ UNIT_TEST(GlyphCacheTest_Main) { - yg::GlyphCache cache; + yg::GlyphCache cache(200000); cache.addFont(GetPlatform().ReadPathForFile("dejavusans.ttf").c_str()); shared_ptr g1 = cache.getGlyph(yg::GlyphKey('#', 40, true)); - g1->dump(GetPlatform().WritablePathForFile("#_mask.png").c_str()); +// g1->dump(GetPlatform().WritablePathForFile("#_mask.png").c_str()); shared_ptr g2 = cache.getGlyph(yg::GlyphKey('#', 40, false)); - g2->dump(GetPlatform().WritablePathForFile("#.png").c_str()); +// g2->dump(GetPlatform().WritablePathForFile("#.png").c_str()); } diff --git a/yg/yg_tests/screengl_test.cpp b/yg/yg_tests/screengl_test.cpp index 0f85cdc426..fca274db3d 100644 --- a/yg/yg_tests/screengl_test.cpp +++ b/yg/yg_tests/screengl_test.cpp @@ -437,6 +437,14 @@ namespace } }; + struct TestDrawEmptySymbol + { + void DoDraw(shared_ptr p) + { + p->drawText(m2::PointD(40, 50), 0, 20, " ", 1); + } + }; + struct TestDrawStringOnString { void DoDraw(shared_ptr p) @@ -690,22 +698,23 @@ namespace }; // UNIT_TEST_GL(TestDrawPolyOverflow); - UNIT_TEST_GL(TestDrawFont); -// UNIT_TEST_GL(TestDrawSingleSymbol); +// UNIT_TEST_GL(TestDrawFont); + UNIT_TEST_GL(TestDrawSingleSymbol); +// UNIT_TEST_GL(TestDrawEmptySymbol); // UNIT_TEST_GL(TestDrawSingleSymbolAndSolidPath); - UNIT_TEST_GL(TestDrawString); - UNIT_TEST_GL(TestDrawStringOnString); - UNIT_TEST_GL(TestDrawFontOnPath); - UNIT_TEST_GL(TestDrawFontOnPathWithOffset); +// UNIT_TEST_GL(TestDrawString); +// UNIT_TEST_GL(TestDrawStringOnString); +// UNIT_TEST_GL(TestDrawFontOnPath); +// UNIT_TEST_GL(TestDrawFontOnPathWithOffset); // UNIT_TEST_GL(TestDrawSGIConvex); // UNIT_TEST_GL(TestDrawPoly); // UNIT_TEST_GL(TestDrawSolidRect); -// UNIT_TEST_GL(TestDrawPathWithSkinPageMiss); +// UNIT_TEST_GL(TestDrawPathWithSkinPageMiss); // UNIT_TEST_GL(TestDrawPathWithOffset); // UNIT_TEST_GL(TestDrawPathJoin); // UNIT_TEST_GL(TestDrawPathSolid); // UNIT_TEST_GL(TestDrawPathSolidWithZ); -// UNIT_TEST_GL(TestDrawPathSolidWithClipRect); +// UNIT_TEST_GL(TestDrawPathSolidWithClipRect); // UNIT_TEST_GL(TestDrawUtilsRect); // UNIT_TEST_GL(TestDrawUtilsRectFilledTexture); } diff --git a/yg/yg_tests/skin_loader_test.cpp b/yg/yg_tests/skin_loader_test.cpp index 72d313efcb..bd0f4e56ea 100644 --- a/yg/yg_tests/skin_loader_test.cpp +++ b/yg/yg_tests/skin_loader_test.cpp @@ -8,6 +8,6 @@ UNIT_TEST(SkinLoaderTest_Main) { GL_TEST_START; - shared_ptr rm(new yg::ResourceManager(1000, 1000, 2, 1000, 1000, 2, 128, 128, 15)); + shared_ptr rm(new yg::ResourceManager(1000, 1000, 2, 1000, 1000, 2, 128, 128, 15, 2000000)); /*yg::Skin * skin = */loadSkin(rm, "basic.skn"); }; diff --git a/yg/yg_tests/skin_test.cpp b/yg/yg_tests/skin_test.cpp index 5774d87cee..8a29eb8c3f 100644 --- a/yg/yg_tests/skin_test.cpp +++ b/yg/yg_tests/skin_test.cpp @@ -9,7 +9,7 @@ UNIT_TEST(SkinTest_Main) { GL_TEST_START; - shared_ptr rm(new yg::ResourceManager(100, 100, 1, 100, 100, 1, 128, 128, 15)); + shared_ptr rm(new yg::ResourceManager(100, 100, 1, 100, 100, 1, 128, 128, 15, 2000000)); yg::Skin * skin = loadSkin(rm, "test.skn"); double p0 [] = {1, 1};