diff --git a/drape/drape_common.pri b/drape/drape_common.pri index d7b2c9a2ab..5533fe3670 100644 --- a/drape/drape_common.pri +++ b/drape/drape_common.pri @@ -32,7 +32,6 @@ SOURCES += \ $$DRAPE_DIR/batcher_helpers.cpp \ $$DRAPE_DIR/overlay_tree.cpp \ $$DRAPE_DIR/font_texture.cpp \ - $$DRAPE_DIR/font_loader.cpp \ $$DRAPE_DIR/texture_set_holder.cpp \ HEADERS += \ @@ -71,4 +70,3 @@ HEADERS += \ $$DRAPE_DIR/batcher_helpers.hpp \ $$DRAPE_DIR/overlay_tree.hpp \ $$DRAPE_DIR/font_texture.hpp \ - $$DRAPE_DIR/font_loader.hpp \ diff --git a/drape/drape_tests/font_loader_tests.cpp b/drape/drape_tests/font_texture_tests.cpp similarity index 100% rename from drape/drape_tests/font_loader_tests.cpp rename to drape/drape_tests/font_texture_tests.cpp diff --git a/drape/font_loader.cpp b/drape/font_loader.cpp deleted file mode 100644 index 6ce51caa54..0000000000 --- a/drape/font_loader.cpp +++ /dev/null @@ -1,147 +0,0 @@ -#include "font_loader.hpp" - -#include "../platform/platform.hpp" - -#include "glfunctions.hpp" -#include "glconstants.hpp" -#include "utils/stb_image.h" - -#include "../std/string.hpp" -#include "../std/map.hpp" -#include "../std/vector.hpp" - -#include -#include -#include - -using boost::gil::gray8_view_t; -using boost::gil::gray8c_view_t; -using boost::gil::interleaved_view; -using boost::gil::subimage_view; -using boost::gil::copy_pixels; -using boost::gil::gray8c_pixel_t; -using boost::gil::gray8_pixel_t; - -namespace -{ - void CreateFontChar(string const & s, FontChar & symbol) - { - vector tokens; - strings::Tokenize(s, "\t", MakeBackInsertFunctor(tokens)); - - strings::to_int(tokens[0].c_str(), symbol.m_unicode); - strings::to_int(tokens[1].c_str(), symbol.m_x); - strings::to_int(tokens[2].c_str(), symbol.m_y); - strings::to_int(tokens[3].c_str(), symbol.m_width); - strings::to_int(tokens[4].c_str(), symbol.m_height); - strings::to_int(tokens[5].c_str(), symbol.m_xOffset); - strings::to_int(tokens[6].c_str(), symbol.m_yOffset); - strings::to_int(tokens[7].c_str(), symbol.m_spacing); - } -} - -void FontLoader::Add(FontChar const & symbol) -{ - m_dictionary.insert(make_pair(symbol.m_unicode, symbol)); -} - -int FontLoader::GetSymbolCoords(FontChar & symbol) const -{ - int block = (symbol.m_y / m_supportedSize) * m_blockCnt + symbol.m_x / m_supportedSize; - symbol.m_blockNum = block; - symbol.m_x %= m_supportedSize; - symbol.m_y %= m_supportedSize; - return block; -} - -int FontLoader::GetBlockByUnicode(int unicode) const -{ - FontChar symbol; - if(GetSymbolByUnicode(unicode, symbol)) - return symbol.m_blockNum; - - return -1; -} - -bool FontLoader::GetSymbolByUnicode(int unicode, FontChar & symbol) const -{ - map::const_iterator itm = m_dictionary.find(unicode); - if(itm == m_dictionary.end()) - return false; - - symbol = itm->second; - return true; -} - -vector FontLoader::Load(string const & path) -{ - m_dictionary.clear(); - vector buffer; - - try - { - ReaderPtr reader = GetPlatform().GetReader(path + ".png"); - uint64_t size = reader.Size(); - buffer.resize(size); - reader.Read(0, &buffer[0], size); - } - catch (RootException & e) - { - LOG(LERROR, (e.what())); - return vector(); - } - - int w, h, bpp; - unsigned char * data = stbi_png_load_from_memory(&buffer[0], buffer.size(), &w, &h, &bpp, 0); - CHECK(bpp == 1, ("WRONG_FONT_TEXTURE_FORMAT")); - - m_realSize = w; - int32_t maxTextureSize = GLFunctions::glGetInteger(gl_const::GLMaxTextureSize); - m_supportedSize = min(m_realSize, maxTextureSize); - m_blockCnt = m_realSize / m_supportedSize; - - vector pages; - pages.resize(m_blockCnt * m_blockCnt); - - buffer.resize(m_supportedSize * m_supportedSize); - gray8c_view_t srcImage = interleaved_view(w, h, (gray8c_pixel_t const *)data, w); - - for(int i = 0 ; i < m_blockCnt ; ++i) - { - for(int j = 0; j < m_blockCnt ; ++j) - { - gray8c_view_t subSrcImage = subimage_view(srcImage, j * m_supportedSize, i * m_supportedSize, m_supportedSize, m_supportedSize); - gray8_view_t dstImage = interleaved_view(m_supportedSize, m_supportedSize, (gray8_pixel_t *)&buffer[0], m_supportedSize); - copy_pixels(subSrcImage, dstImage); - pages[i * m_blockCnt + j].Load(m_supportedSize, &buffer[0], i * m_blockCnt + j); - } - } - stbi_image_free(data); - buffer.clear(); - - string s; - try - { - ReaderPtr reader = GetPlatform().GetReader(path + ".txt"); - reader.ReadAsString(s); - } - catch (RootException & e) - { - LOG(LERROR, (e.what())); - pages.clear(); - return vector(); - } - - vector tokens; - strings::Tokenize(s, "\n", MakeBackInsertFunctor(tokens)); - - for(int i = 0; i < tokens.size() ; ++i) - { - FontChar symbol; - CreateFontChar(tokens[i], symbol); - pages[GetSymbolCoords(symbol)].Add(symbol); - Add(symbol); - } - - return pages; -} diff --git a/drape/font_loader.hpp b/drape/font_loader.hpp deleted file mode 100644 index dd2b8c49bd..0000000000 --- a/drape/font_loader.hpp +++ /dev/null @@ -1,42 +0,0 @@ -#pragma once - -#include "texture_font.hpp" - -#include "../platform/platform.hpp" - -#include "../base/string_utils.hpp" -#include "../base/stl_add.hpp" - -#include "../std/map.hpp" - -struct FontChar -{ -public: - int m_unicode; - int m_x; - int m_y; - int m_height; - int m_width; - int m_xOffset; - int m_yOffset; - int m_spacing; - int m_blockNum; -}; - -class TextureFont; -class FontLoader -{ -private: - int m_realSize; - int m_supportedSize; - int m_blockCnt; - - map m_dictionary; - -public: - void Add(FontChar const & symbol); - int GetSymbolCoords(FontChar & symbol) const; - int GetBlockByUnicode(int unicode) const; - bool GetSymbolByUnicode(int unicode, FontChar & symbol) const; - vector Load(string const & path); -}; diff --git a/drape/font_texture.cpp b/drape/font_texture.cpp new file mode 100644 index 0000000000..bc5b78e99f --- /dev/null +++ b/drape/font_texture.cpp @@ -0,0 +1,253 @@ +#include "font_texture.hpp" +#include "pointers.hpp" + +#include "../platform/platform.hpp" +#include "../coding/reader.hpp" +#include "../coding/stb_image.h" + +#include "../base/logging.hpp" +#include "../base/string_utils.hpp" +#include "../base/stl_add.hpp" + +#include "../std/string.hpp" +#include "../std/vector.hpp" +#include "../std/map.hpp" +#include "../std/bind.hpp" + +#include +#include + +using boost::gil::gray8c_pixel_t; +using boost::gil::gray8_pixel_t; +using boost::gil::gray8c_view_t; +using boost::gil::gray8_view_t; +using boost::gil::interleaved_view; +using boost::gil::subimage_view; +using boost::gil::copy_pixels; + +typedef gray8_view_t view_t; +typedef gray8c_view_t const_view_t; +typedef gray8_pixel_t pixel_t; +typedef gray8c_pixel_t const_pixel_t; + +FontTexture::GlyphInfo::GlyphInfo(m2::RectF const & texRect, float xOffset, + float yOffset, float advance) + : ResourceInfo(texRect) + , m_xOffset(xOffset) + , m_yOffset(yOffset) + , m_advance(advance) +{ +} + +void FontTexture::GlyphInfo::GetMetrics(float & xOffset, float & yOffset, float & advance) const +{ + xOffset = m_xOffset; + yOffset = m_yOffset; + advance = m_advance; +} + +//////////////////////////////////////////////////////////////////////// + +Texture::ResourceInfo const * FontTexture::FindResource(Texture::Key const & key) const +{ + if (key.GetType() != Texture::Glyph) + return false; + + int unicodePoint = static_cast(key).GetUnicodePoint(); + glyph_map_t::const_iterator it = m_glyphs.find(unicodePoint); + if (it == m_glyphs.end()) + return NULL; + + return &it->second; +} + +void FontTexture::Add(int unicodePoint, GlyphInfo const & glyphInfo) +{ + m_glyphs.insert(make_pair(unicodePoint, glyphInfo)); +} + +namespace +{ + class Grid + { + typedef pair region_t; + typedef vector regions_t; + + public: + void CutTexture(vector const & image, uint32_t width, uint32_t height) + { + uint32_t maxTextureSize = Texture::GetMaxTextureSize(); + + if (width <= maxTextureSize && width == height) + SingleTexture(image, width, height); + else + { + const_view_t srcView = interleaved_view(width, height, (const_pixel_t *)&image[0], width); + uint32_t baseSize = maxTextureSize; + vector notProcessed; + notProcessed.push_back(m2::RectU(0, 0, width, height)); + while (!notProcessed.empty()) + { + vector outRects; + for (size_t i = 0; i < notProcessed.size(); ++i) + CutTextureBySize(srcView, notProcessed[i], baseSize, outRects); + + baseSize >>= 1; + notProcessed = outRects; + } + } + } + + void ParseMetrics(string const & fileData) + { + vector lines; + strings::Tokenize(fileData, "\n", MakeBackInsertFunctor(lines)); + for (size_t i = 0; i < lines.size(); ++i) + { + vector metrics; + strings::Tokenize(lines[i], "\t", MakeBackInsertFunctor(metrics)); + ASSERT(metrics.size() == 8, ()); + + int32_t unicodePoint; + int32_t x, y, w, h; + double xoff, yoff, advance; + + strings::to_int(metrics[0], unicodePoint); + strings::to_int(metrics[1], x); + strings::to_int(metrics[2], y); + strings::to_int(metrics[3], w); + strings::to_int(metrics[4], h); + strings::to_double(metrics[5], xoff); + strings::to_double(metrics[6], yoff); + strings::to_double(metrics[7], advance); + + m2::PointU centerPoint(x + w / 2, y + h / 2); + region_t region = GetTexture(centerPoint); + FontTexture * texture = region.second; + m2::RectU rect = region.first; + m2::RectF texRect(texture->GetS(x - rect.minX()), texture->GetT(y - rect.minY()), + texture->GetS(x + w - rect.minX()), texture->GetT(y + h - rect.minY())); + texture->Add(unicodePoint, FontTexture::GlyphInfo(texRect, xoff, yoff, advance)); + } + } + + void GetTextures(vector > & textures) + { + textures.reserve(m_regions.size()); + for (size_t i = 0; i < m_regions.size(); ++i) + textures.push_back(MovePointer(m_regions[i].second)); + } + + private: + void SingleTexture(vector const & image, uint32_t width, uint32_t height) + { + FontTexture * texture = new FontTexture(); + texture->Create(width, height, Texture::ALPHA, MakeStackRefPointer((void *)&image[0])); + m_regions.push_back(make_pair(m2::RectU(0, 0, width, height), texture)); + } + + void CutTextureBySize(const_view_t const & image, m2::RectU const & fullRect, + uint32_t cutSize, vector & notProcessedRects) + { + uint32_t fullTexInWidth = fullRect.SizeX() / cutSize; + uint32_t fullTexInHeight = fullRect.SizeY() / cutSize; + + if (fullTexInWidth == 0 || fullTexInHeight == 0) + { + notProcessedRects.push_back(fullRect); + return; + } + + vector regionImage(cutSize * cutSize, 0); + for (uint32_t dy = 0; dy < fullTexInHeight; ++dy) + { + for (uint32_t dx = 0; dx < fullTexInWidth; ++dx) + { + uint32_t pxDx = dx * cutSize + fullRect.minX(); + uint32_t pxDy = dy * cutSize + fullRect.minY(); + const_view_t subView = subimage_view(image, pxDx, pxDy, cutSize, cutSize); + + view_t dstView = interleaved_view(cutSize, cutSize, + (pixel_t *)®ionImage[0], cutSize); + + copy_pixels(subView, dstView); + FontTexture * texture = new FontTexture(); + texture->Create(cutSize, cutSize, Texture::ALPHA, + MakeStackRefPointer(®ionImage[0])); + + m_regions.push_back(make_pair(m2::RectU(pxDx, pxDy, + pxDx + cutSize, + pxDy + cutSize), + texture)); + } + } + + uint32_t downBorder = fullTexInHeight * cutSize; + uint32_t rightBorder = fullTexInWidth * cutSize; + if (rightBorder == fullRect.SizeX()) + { + ASSERT(downBorder == fullRect.SizeY(), ()); + return; + } + + notProcessedRects.push_back(m2::RectU(rightBorder, 0, fullRect.maxX(), downBorder)); + + if (downBorder == fullRect.SizeY()) + return; + + notProcessedRects.push_back(m2::RectU(0, downBorder, rightBorder, fullRect.maxY())); + } + + private: + region_t GetTexture(m2::PointU const & px) + { + regions_t::const_iterator it = find_if(m_regions.begin(), m_regions.end(), + bind(&m2::RectU::IsPointInside, + bind(®ion_t::first, _1), px)); + + ASSERT(it != m_regions.end(), ()); + return *it; + } + + private: + regions_t m_regions; + }; +} + +void LoadFont(string const & resourcePrefix, vector > & textures) +{ + string metrics; + int w, h, channelCount; + vector imageData; + + try + { + { + ReaderPtr reader = GetPlatform().GetReader(resourcePrefix + ".png"); + imageData.resize(reader.Size()); + reader.Read(0, &imageData[0], imageData.size()); + + unsigned char * img = stbi_png_load_from_memory(&imageData[0], imageData.size(), + &w, &h, &channelCount, 0); + CHECK(channelCount == 1, ("Incorrect font texture format")); + + imageData.resize(w * h); + memcpy(&imageData[0], img, w * h); + stbi_image_free(img); + } + + { + ReaderPtr reader = GetPlatform().GetReader(resourcePrefix + ".fdf"); + reader.ReadAsString(metrics); + } + } + catch (RootException & e) + { + LOG(LERROR, (e.what())); + } + + Grid grid; + grid.CutTexture(imageData, w, h); + grid.ParseMetrics(metrics); + grid.GetTextures(textures); +} diff --git a/drape/font_texture.hpp b/drape/font_texture.hpp new file mode 100644 index 0000000000..9edb92fc68 --- /dev/null +++ b/drape/font_texture.hpp @@ -0,0 +1,49 @@ +#pragma once + +#include "pointers.hpp" +#include "texture.hpp" + +#include "../std/map.hpp" +#include "../std/vector.hpp" +#include "../std/string.hpp" + +class FontTexture : public Texture +{ +public: + class GlyphKey : public Key + { + public: + GlyphKey(int32_t unicode) : m_unicode(unicode) {} + + ResourceType GetType() const { return Texture::Glyph; } + int32_t GetUnicodePoint() const { return m_unicode; } + + private: + int32_t m_unicode; + }; + + class GlyphInfo : public ResourceInfo + { + public: + GlyphInfo(m2::RectF const & texRect, float xOffset, + float yOffset, float advance); + + virtual ResourceType GetType() const { return Texture::Glyph; } + void GetMetrics(float & xOffset, float & yOffset, float & advance) const; + + private: + float m_xOffset, m_yOffset; + float m_advance; + }; + +public: + ResourceInfo const * FindResource(Key const & key) const; + + void Add(int unicodePoint, GlyphInfo const & glyphInfo); + +private: + typedef map glyph_map_t; + glyph_map_t m_glyphs; +}; + +void LoadFont(string const & resourcePrefix, vector > & textures); diff --git a/drape/texture.cpp b/drape/texture.cpp index baad7fe2ac..06868277e4 100644 --- a/drape/texture.cpp +++ b/drape/texture.cpp @@ -113,6 +113,11 @@ void Texture::Bind() const GLFunctions::glBindTexture(GetID()); } +uint32_t Texture::GetMaxTextureSize() +{ + return GLFunctions::glGetInteger(gl_const::GLMaxTextureSize); +} + void Texture::UnpackFormat(Texture::TextureFormat format, glConst & layout, glConst & pixelType) { bool requiredFormat = GLExtensionsList::Instance().IsSupported(GLExtensionsList::RequiredInternalFormat); diff --git a/drape/texture.hpp b/drape/texture.hpp index 94cffdc21a..fdf7f5073a 100644 --- a/drape/texture.hpp +++ b/drape/texture.hpp @@ -28,6 +28,7 @@ public: class Key { public: + virtual ~Key() {} virtual ResourceType GetType() const = 0; }; @@ -35,6 +36,7 @@ public: { public: ResourceInfo(m2::RectF const & texRect); + virtual ~ResourceInfo() {} virtual ResourceType GetType() const = 0; m2::RectF const & GetTexRect() const; @@ -63,6 +65,8 @@ public: void Bind() const; + static uint32_t GetMaxTextureSize(); + private: void UnpackFormat(TextureFormat format, glConst & layout, glConst & pixelType); int32_t GetID() const; diff --git a/drape/texture_font.cpp b/drape/texture_font.cpp deleted file mode 100644 index 8dc8bac2a4..0000000000 --- a/drape/texture_font.cpp +++ /dev/null @@ -1,46 +0,0 @@ -#include "texture_font.hpp" - -bool TextureFont::FindResource(Texture::Key const & key, m2::RectF & texRect, m2::PointU & pixelSize) const -{ - if (key.GetType() != Texture::Key::Font) - return false; - - map::const_iterator itm = m_chars.find(static_cast(key).GetUnicode()); - if(itm == m_chars.end()) - return false; - - FontChar const & symbol = itm->second; - - pixelSize.x = symbol.m_width; - pixelSize.y = symbol.m_height; - - float x_start = (float)symbol.m_x / (float)GetWidth(); - float y_start = (float)symbol.m_y / (float)GetHeight(); - float dx = (float)symbol.m_width / (float)GetWidth(); - float dy = (float)symbol.m_height / (float)GetHeight(); - - texRect = m2::RectF(x_start, y_start, x_start + dx, y_start + dy); - - return true; -} - -bool TextureFont::GetSymbolByUnicode(int unicode, FontChar & symbol) const -{ - map::const_iterator itm = m_chars.find(unicode); - if(itm == m_chars.end()) - return false; - - symbol = itm->second; - return true; -} - -void TextureFont::Load(int size, void *data, int32_t blockNum) -{ - m_blockNum = blockNum; - Create(size, size, Texture::ALPHA, MakeStackRefPointer(data)); -} - -void TextureFont::Add(FontChar const & symbol) -{ - m_chars.insert(make_pair(symbol.m_unicode, symbol)); -} diff --git a/drape/texture_font.hpp b/drape/texture_font.hpp deleted file mode 100644 index 8f570730ea..0000000000 --- a/drape/texture_font.hpp +++ /dev/null @@ -1,34 +0,0 @@ -#pragma once - -#include "texture.hpp" -#include "font_loader.hpp" -#include "../std/map.hpp" - -using namespace std; - -struct FontChar; -class TextureFont : public Texture -{ -public: - class FontKey : public Key - { - public: - FontKey(int32_t unicode) : m_unicode(unicode) {} - virtual Type GetType() const { return Texture::Key::Font; } - int32_t GetUnicode() const { return m_unicode; } - - private: - int32_t m_unicode; - }; - -public: - bool FindResource(Key const & key, m2::RectF & texRect, m2::PointU & pixelSize) const; - void Load(int size, void * data, int32_t blockNum); - void Add(FontChar const & letter); - - bool GetSymbolByUnicode(int unicode, FontChar & symbol) const; - -private: - map m_chars; - int32_t m_blockNum; -}; diff --git a/drape/texture_manager.cpp b/drape/texture_manager.cpp index ca9e9bc617..151dbb98f8 100644 --- a/drape/texture_manager.cpp +++ b/drape/texture_manager.cpp @@ -1,5 +1,6 @@ #include "texture_manager.hpp" #include "symbols_texture.hpp" +#include "font_texture.hpp" #include "glfunctions.hpp" @@ -73,20 +74,36 @@ private: uint32_t m_maxSize; }; -void TextureManager::Init(string const & resourcePostfix) +void TextureManager::Init(string const & resourcePrefix) { // in shader we handle only 8 textures m_maxTextureBlocks = min(8, GLFunctions::glGetInteger(gl_const::GLMaxFragmentTextures)); SymbolsTexture * symbols = new SymbolsTexture(); - symbols->Load(my::JoinFoldersToPath(string("resources-") + resourcePostfix, "symbols")); + symbols->Load(my::JoinFoldersToPath(string("resources-") + resourcePrefix, "symbols")); - m_textures.Reset(new TextureSet(m_maxTextureBlocks)); - m_textures->AddTexture(MovePointer(symbols)); + TextureSet * defaultSet = new TextureSet(m_maxTextureBlocks); + defaultSet->AddTexture(MovePointer(symbols)); + + m_textures.push_back(MasterPointer(defaultSet)); + + vector > tempTextures; + LoadFont(string("resources"), tempTextures); + for (size_t i = 0; i < tempTextures.size(); ++i) + { + RefPointer set = m_textures.back().GetRefPointer(); + if (set->IsFull()) + { + m_textures.push_back(MasterPointer(new TextureSet(m_maxTextureBlocks))); + set = m_textures.back().GetRefPointer(); + } + + set->AddTexture(tempTextures[i]); + } } void TextureManager::Release() { - m_textures.Destroy(); + DeleteRange(m_textures, MasterPointerDeleter()); } void TextureManager::GetSymbolRegion(string const & symbolName, SymbolRegion & region) const @@ -94,27 +111,41 @@ void TextureManager::GetSymbolRegion(string const & symbolName, SymbolRegion & r SymbolsTexture::SymbolKey key(symbolName); TextureNode node; node.m_textureSet = 0; - Texture::ResourceInfo const * info = m_textures->FindResource(key, node); + Texture::ResourceInfo const * info = m_textures[0]->FindResource(key, node); ASSERT(node.m_textureOffset != -1, ()); region.SetResourceInfo(info); region.SetTextureNode(node); } -void TextureManager::GetGlyphRegion(strings::UniChar charCode, GlyphRegion & region) const +bool TextureManager::GetGlyphRegion(strings::UniChar charCode, GlyphRegion & region) const { - // todo; + FontTexture::GlyphKey key(charCode); + for (size_t i = 0; i < m_textures.size(); ++i) + { + TextureNode node; + Texture::ResourceInfo const * info = m_textures[i]->FindResource(key, node); + if (info != NULL) + { + node.m_textureSet = i; + region.SetTextureNode(node); + region.SetResourceInfo(info); + return true; + } + } + + return false; } void TextureManager::BindTextureSet(uint32_t textureSet) const { - ASSERT_LESS(textureSet, 1, ()); // TODO replace 1 to m_textureSets.size() - m_textures->BindTextureSet(); + ASSERT_LESS(textureSet, m_textures.size(), ()); + m_textures[textureSet]->BindTextureSet(); } uint32_t TextureManager::GetTextureCount(uint32_t textureSet) const { - ASSERT_LESS(textureSet, 1, ()); // TODO replace 1 to m_textureSets.size() - return m_textures->GetSize(); + ASSERT_LESS(textureSet, m_textures.size(), ()); + return m_textures[textureSet]->GetSize(); } diff --git a/drape/texture_manager.hpp b/drape/texture_manager.hpp index e62acedd69..a039b845fa 100644 --- a/drape/texture_manager.hpp +++ b/drape/texture_manager.hpp @@ -12,14 +12,14 @@ public: void Init(string const & resourcePrefix); void Release(); virtual void GetSymbolRegion(string const & symbolName, SymbolRegion & region) const; - virtual void GetGlyphRegion(strings::UniChar charCode, GlyphRegion & region) const; + virtual bool GetGlyphRegion(strings::UniChar charCode, GlyphRegion & region) const; void BindTextureSet(uint32_t textureSet) const; uint32_t GetTextureCount(uint32_t textureSet) const; private: class TextureSet; - MasterPointer m_textures; + vector > m_textures; uint32_t m_maxTextureBlocks; }; diff --git a/drape/texture_set_holder.hpp b/drape/texture_set_holder.hpp index d4db3bb035..19ad6e9d8d 100644 --- a/drape/texture_set_holder.hpp +++ b/drape/texture_set_holder.hpp @@ -57,5 +57,5 @@ public: }; virtual void GetSymbolRegion(string const & symbolName, SymbolRegion & region) const = 0; - virtual void GetGlyphRegion(strings::UniChar charCode, GlyphRegion & region) const = 0; + virtual bool GetGlyphRegion(strings::UniChar charCode, GlyphRegion & region) const = 0; };